Taner Sener 7 lat temu
rodzic
commit
4b4492f87c
100 zmienionych plików z 9689 dodań i 13 usunięć
  1. 165 1
      README.md
  2. 0 11
      example/README.md
  3. 1 1
      example/pubspec.yaml
  4. 2 0
      flutter_ffmpeg.iml
  5. 7 0
      packages/flutter_ffmpeg_audio/.gitignore
  6. 10 0
      packages/flutter_ffmpeg_audio/.metadata
  7. 165 0
      packages/flutter_ffmpeg_audio/LICENSE
  8. 8 0
      packages/flutter_ffmpeg_audio/android/.gitignore
  9. 38 0
      packages/flutter_ffmpeg_audio/android/build.gradle
  10. 1 0
      packages/flutter_ffmpeg_audio/android/gradle.properties
  11. 1 0
      packages/flutter_ffmpeg_audio/android/settings.gradle
  12. 3 0
      packages/flutter_ffmpeg_audio/android/src/main/AndroidManifest.xml
  13. 66 0
      packages/flutter_ffmpeg_audio/android/src/main/java/com/arthenica/flutter/ffmpeg/FlutterFFmpegExecuteAsyncArgumentsTask.java
  14. 72 0
      packages/flutter_ffmpeg_audio/android/src/main/java/com/arthenica/flutter/ffmpeg/FlutterFFmpegExecuteAsyncCommandTask.java
  15. 70 0
      packages/flutter_ffmpeg_audio/android/src/main/java/com/arthenica/flutter/ffmpeg/FlutterFFmpegGetMediaInformationAsyncTask.java
  16. 436 0
      packages/flutter_ffmpeg_audio/android/src/main/java/com/arthenica/flutter/ffmpeg/FlutterFFmpegPlugin.java
  17. 36 0
      packages/flutter_ffmpeg_audio/ios/.gitignore
  18. 0 0
      packages/flutter_ffmpeg_audio/ios/Assets/.gitkeep
  19. 27 0
      packages/flutter_ffmpeg_audio/ios/Classes/FlutterFFmpegPlugin.h
  20. 387 0
      packages/flutter_ffmpeg_audio/ios/Classes/FlutterFfmpegPlugin.m
  21. 22 0
      packages/flutter_ffmpeg_audio/ios/flutter_ffmpeg.podspec
  22. 352 0
      packages/flutter_ffmpeg_audio/lib/flutter_ffmpeg.dart
  23. 76 0
      packages/flutter_ffmpeg_audio/lib/log_level.dart
  24. 17 0
      packages/flutter_ffmpeg_audio/pubspec.yaml
  25. 7 0
      packages/flutter_ffmpeg_full-gpl/.gitignore
  26. 10 0
      packages/flutter_ffmpeg_full-gpl/.metadata
  27. 674 0
      packages/flutter_ffmpeg_full-gpl/LICENSE
  28. 8 0
      packages/flutter_ffmpeg_full-gpl/android/.gitignore
  29. 38 0
      packages/flutter_ffmpeg_full-gpl/android/build.gradle
  30. 1 0
      packages/flutter_ffmpeg_full-gpl/android/gradle.properties
  31. 1 0
      packages/flutter_ffmpeg_full-gpl/android/settings.gradle
  32. 3 0
      packages/flutter_ffmpeg_full-gpl/android/src/main/AndroidManifest.xml
  33. 66 0
      packages/flutter_ffmpeg_full-gpl/android/src/main/java/com/arthenica/flutter/ffmpeg/FlutterFFmpegExecuteAsyncArgumentsTask.java
  34. 72 0
      packages/flutter_ffmpeg_full-gpl/android/src/main/java/com/arthenica/flutter/ffmpeg/FlutterFFmpegExecuteAsyncCommandTask.java
  35. 70 0
      packages/flutter_ffmpeg_full-gpl/android/src/main/java/com/arthenica/flutter/ffmpeg/FlutterFFmpegGetMediaInformationAsyncTask.java
  36. 436 0
      packages/flutter_ffmpeg_full-gpl/android/src/main/java/com/arthenica/flutter/ffmpeg/FlutterFFmpegPlugin.java
  37. 36 0
      packages/flutter_ffmpeg_full-gpl/ios/.gitignore
  38. 0 0
      packages/flutter_ffmpeg_full-gpl/ios/Assets/.gitkeep
  39. 27 0
      packages/flutter_ffmpeg_full-gpl/ios/Classes/FlutterFFmpegPlugin.h
  40. 387 0
      packages/flutter_ffmpeg_full-gpl/ios/Classes/FlutterFfmpegPlugin.m
  41. 22 0
      packages/flutter_ffmpeg_full-gpl/ios/flutter_ffmpeg.podspec
  42. 352 0
      packages/flutter_ffmpeg_full-gpl/lib/flutter_ffmpeg.dart
  43. 76 0
      packages/flutter_ffmpeg_full-gpl/lib/log_level.dart
  44. 17 0
      packages/flutter_ffmpeg_full-gpl/pubspec.yaml
  45. 7 0
      packages/flutter_ffmpeg_full/.gitignore
  46. 10 0
      packages/flutter_ffmpeg_full/.metadata
  47. 165 0
      packages/flutter_ffmpeg_full/LICENSE
  48. 8 0
      packages/flutter_ffmpeg_full/android/.gitignore
  49. 38 0
      packages/flutter_ffmpeg_full/android/build.gradle
  50. 1 0
      packages/flutter_ffmpeg_full/android/gradle.properties
  51. 1 0
      packages/flutter_ffmpeg_full/android/settings.gradle
  52. 3 0
      packages/flutter_ffmpeg_full/android/src/main/AndroidManifest.xml
  53. 66 0
      packages/flutter_ffmpeg_full/android/src/main/java/com/arthenica/flutter/ffmpeg/FlutterFFmpegExecuteAsyncArgumentsTask.java
  54. 72 0
      packages/flutter_ffmpeg_full/android/src/main/java/com/arthenica/flutter/ffmpeg/FlutterFFmpegExecuteAsyncCommandTask.java
  55. 70 0
      packages/flutter_ffmpeg_full/android/src/main/java/com/arthenica/flutter/ffmpeg/FlutterFFmpegGetMediaInformationAsyncTask.java
  56. 436 0
      packages/flutter_ffmpeg_full/android/src/main/java/com/arthenica/flutter/ffmpeg/FlutterFFmpegPlugin.java
  57. 36 0
      packages/flutter_ffmpeg_full/ios/.gitignore
  58. 0 0
      packages/flutter_ffmpeg_full/ios/Assets/.gitkeep
  59. 27 0
      packages/flutter_ffmpeg_full/ios/Classes/FlutterFFmpegPlugin.h
  60. 387 0
      packages/flutter_ffmpeg_full/ios/Classes/FlutterFfmpegPlugin.m
  61. 22 0
      packages/flutter_ffmpeg_full/ios/flutter_ffmpeg.podspec
  62. 352 0
      packages/flutter_ffmpeg_full/lib/flutter_ffmpeg.dart
  63. 76 0
      packages/flutter_ffmpeg_full/lib/log_level.dart
  64. 17 0
      packages/flutter_ffmpeg_full/pubspec.yaml
  65. 7 0
      packages/flutter_ffmpeg_https-gpl/.gitignore
  66. 10 0
      packages/flutter_ffmpeg_https-gpl/.metadata
  67. 674 0
      packages/flutter_ffmpeg_https-gpl/LICENSE
  68. 8 0
      packages/flutter_ffmpeg_https-gpl/android/.gitignore
  69. 38 0
      packages/flutter_ffmpeg_https-gpl/android/build.gradle
  70. 1 0
      packages/flutter_ffmpeg_https-gpl/android/gradle.properties
  71. 1 0
      packages/flutter_ffmpeg_https-gpl/android/settings.gradle
  72. 3 0
      packages/flutter_ffmpeg_https-gpl/android/src/main/AndroidManifest.xml
  73. 66 0
      packages/flutter_ffmpeg_https-gpl/android/src/main/java/com/arthenica/flutter/ffmpeg/FlutterFFmpegExecuteAsyncArgumentsTask.java
  74. 72 0
      packages/flutter_ffmpeg_https-gpl/android/src/main/java/com/arthenica/flutter/ffmpeg/FlutterFFmpegExecuteAsyncCommandTask.java
  75. 70 0
      packages/flutter_ffmpeg_https-gpl/android/src/main/java/com/arthenica/flutter/ffmpeg/FlutterFFmpegGetMediaInformationAsyncTask.java
  76. 436 0
      packages/flutter_ffmpeg_https-gpl/android/src/main/java/com/arthenica/flutter/ffmpeg/FlutterFFmpegPlugin.java
  77. 36 0
      packages/flutter_ffmpeg_https-gpl/ios/.gitignore
  78. 0 0
      packages/flutter_ffmpeg_https-gpl/ios/Assets/.gitkeep
  79. 27 0
      packages/flutter_ffmpeg_https-gpl/ios/Classes/FlutterFFmpegPlugin.h
  80. 387 0
      packages/flutter_ffmpeg_https-gpl/ios/Classes/FlutterFfmpegPlugin.m
  81. 22 0
      packages/flutter_ffmpeg_https-gpl/ios/flutter_ffmpeg.podspec
  82. 352 0
      packages/flutter_ffmpeg_https-gpl/lib/flutter_ffmpeg.dart
  83. 76 0
      packages/flutter_ffmpeg_https-gpl/lib/log_level.dart
  84. 17 0
      packages/flutter_ffmpeg_https-gpl/pubspec.yaml
  85. 7 0
      packages/flutter_ffmpeg_https/.gitignore
  86. 10 0
      packages/flutter_ffmpeg_https/.metadata
  87. 165 0
      packages/flutter_ffmpeg_https/LICENSE
  88. 8 0
      packages/flutter_ffmpeg_https/android/.gitignore
  89. 38 0
      packages/flutter_ffmpeg_https/android/build.gradle
  90. 1 0
      packages/flutter_ffmpeg_https/android/gradle.properties
  91. 1 0
      packages/flutter_ffmpeg_https/android/settings.gradle
  92. 3 0
      packages/flutter_ffmpeg_https/android/src/main/AndroidManifest.xml
  93. 66 0
      packages/flutter_ffmpeg_https/android/src/main/java/com/arthenica/flutter/ffmpeg/FlutterFFmpegExecuteAsyncArgumentsTask.java
  94. 72 0
      packages/flutter_ffmpeg_https/android/src/main/java/com/arthenica/flutter/ffmpeg/FlutterFFmpegExecuteAsyncCommandTask.java
  95. 70 0
      packages/flutter_ffmpeg_https/android/src/main/java/com/arthenica/flutter/ffmpeg/FlutterFFmpegGetMediaInformationAsyncTask.java
  96. 436 0
      packages/flutter_ffmpeg_https/android/src/main/java/com/arthenica/flutter/ffmpeg/FlutterFFmpegPlugin.java
  97. 36 0
      packages/flutter_ffmpeg_https/ios/.gitignore
  98. 0 0
      packages/flutter_ffmpeg_https/ios/Assets/.gitkeep
  99. 27 0
      packages/flutter_ffmpeg_https/ios/Classes/FlutterFFmpegPlugin.h
  100. 387 0
      packages/flutter_ffmpeg_https/ios/Classes/FlutterFfmpegPlugin.m

+ 165 - 1
README.md

@@ -1,3 +1,167 @@
 # flutter_ffmpeg
 
-FFmpeg plugin for Flutter. Supports iOS and Android.
+FFmpeg plugin for Flutter. Supports iOS and Android.
+
+### 1. Features
+- Based on MobileFFmpeg
+- Supports
+    - Both Android and IOS
+    - FFmpeg `v4.2-dev-x` (master) releases
+    - `arm-v7a`, `arm-v7a-neon`, `arm64-v8a`, `x86` and `x86_64` architectures on Android
+    - `armv7`, `armv7s`, `arm64`, `arm64e`, `i386` and `x86_64` architectures on IOS
+    - 24 external libraries
+
+        `fontconfig`, `freetype`, `fribidi`, `gmp`, `gnutls`, `kvazaar`, `lame`, `libaom`, `libass`, `libiconv`, `libilbc`, `libtheora`, `libvorbis`, `libvpx`, `libwebp`, `libxml2`, `opencore-amr`, `opus`, `shine`, `snappy`, `soxr`, `speex`, `twolame`, `wavpack`
+
+    - 4 external libraries with GPL license
+
+        `vid.stab`, `x264`, `x265`, `xvidcore`
+
+    - `zlib` and `MediaCodec` Android system libraries
+    - `bzip2`, `zlib` IOS system libraries and `AudioToolbox`, `CoreImage`, `VideoToolbox`, `AVFoundation` IOS system frameworks
+
+- Licensed under LGPL 3.0, can be customized to support GPL v3.0
+- Includes eight different packages with different external libraries enabled in FFmpeg
+
+<table>
+<thead>
+<tr>
+<th align="center"></th>
+<th align="center">min</th>
+<th align="center">min-gpl</th>
+<th align="center">https</th>
+<th align="center">https-gpl</th>
+<th align="center">audio</th>
+<th align="center">video</th>
+<th align="center">full</th>
+<th align="center">full-gpl</th>
+</tr>
+</thead>
+<tbody>
+<tr>
+<td align="center"><sup>external libraries</sup></td>
+<td align="center">-</td>
+<td align="center"><sup>vid.stab</sup><br><sup>x264</sup><br><sup>x265</sup><br><sup>xvidcore</sup></td>
+<td align="center"><sup>gmp</sup><br><sup>gnutls</sup></td>
+<td align="center"><sup>gmp</sup><br><sup>gnutls</sup><br><sup>vid.stab</sup><br><sup>x264</sup><br><sup>x265</sup><br><sup>xvidcore</sup></td>
+<td align="center"><sup>lame</sup><br><sup>libilbc</sup><br><sup>libvorbis</sup><br><sup>opencore-amr</sup><br><sup>opus</sup><br><sup>shine</sup><br><sup>soxr</sup><br><sup>speex</sup><br><sup>twolame</sup><br><sup>wavpack</sup></td>
+<td align="center"><sup>fontconfig</sup><br><sup>freetype</sup><br><sup>fribidi</sup><br><sup>kvazaar</sup><br><sup>libaom</sup><br><sup>libass</sup><br><sup>libiconv</sup><br><sup>libtheora</sup><br><sup>libvpx</sup><br><sup>libwebp</sup><br><sup>snappy</sup></td>
+<td align="center"><sup>fontconfig</sup><br><sup>freetype</sup><br><sup>fribidi</sup><br><sup>gmp</sup><br><sup>gnutls</sup><br><sup>kvazaar</sup><br><sup>lame</sup><br><sup>libaom</sup><br><sup>libass</sup><br><sup>libiconv</sup><br><sup>libilbc</sup><br><sup>libtheora</sup><br><sup>libvorbis</sup><br><sup>libvpx</sup><br><sup>libwebp</sup><br><sup>libxml2</sup><br><sup>opencore-amr</sup><br><sup>opus</sup><br><sup>shine</sup><br><sup>snappy</sup><br><sup>soxr</sup><br><sup>speex</sup><br><sup>twolame</sup><br><sup>wavpack</sup></td>
+<td align="center"><sup>fontconfig</sup><br><sup>freetype</sup><br><sup>fribidi</sup><br><sup>gmp</sup><br><sup>gnutls</sup><br><sup>kvazaar</sup><br><sup>lame</sup><br><sup>libaom</sup><br><sup>libass</sup><br><sup>libiconv</sup><br><sup>libilbc</sup><br><sup>libtheora</sup><br><sup>libvorbis</sup><br><sup>libvpx</sup><br><sup>libwebp</sup><br><sup>libxml2</sup><br><sup>opencore-amr</sup><br><sup>opus</sup><br><sup>shine</sup><br><sup>snappy</sup><br><sup>soxr</sup><br><sup>speex</sup><br><sup>twolame</sup><br><sup>vid.stab</sup><br><sup>wavpack</sup><br><sup>x264</sup><br><sup>x265</sup><br><sup>xvidcore</sup></td>
+</tr>
+<tr>
+<td align="center"><sup>android system libraries</sup></td>
+<td align="center" colspan=8><sup>zlib</sup><br><sup>MediaCodec</sup></td>
+</tr>
+<tr>
+<td align="center"><sup>ios system libraries</sup></td>
+<td align="center" colspan=8><sup>zlib</sup><br><sup>AudioToolbox</sup><br><sup>AVFoundation</sup><br><sup>CoreImage</sup><br><sup>VideoToolbox</sup><br><sup>bzip2</sup></td>
+</tr>
+</tbody>
+</table>
+
+### 2. Installation
+
+Add `flutter_ffmpeg` as a [dependency in your pubspec.yaml file](https://flutter.io/platform-plugins).
+
+#### 2.1 Packages
+
+Default installation of `FlutterFFmpeg` enables the default package, which is based on `https` package. It is possible
+to enable other installed packages using the following steps.
+
+    ```
+
+    ```
+
+### 3. Using
+
+1. Execute commands.
+    ```
+    import 'package:flutter_ffmpeg/flutter_ffmpeg.dart';
+
+    final FlutterFFmpeg _flutterFFmpeg = new FlutterFFmpeg();
+
+    _flutterFFmpeg.execute("-i file1.mp4 -c:v mpeg4 file2.mp4").then((rc) => print("FFmpeg process exited with rc $rc"));
+    ```
+
+2. Check execution output.
+    ```
+    _flutterFFmpeg.getLastReturnCode().then((rc) => print("Last rc: $rc"));
+    _flutterFFmpeg.getLastCommandOutput().then((output) => print("Last command output: $output"));
+    ```
+
+3. Stop an ongoing operation.
+    ```
+    _flutterFFmpeg.cancel();
+    ```
+
+4. Get media information for a file.
+    ```
+    _flutterFFmpeg.getMediaInformation('<file path or uri>').then((info) => print(info));
+    ```
+
+5. List enabled external libraries.
+    ```
+    _flutterFFmpeg.getExternalLibraries().then((packageList) {
+         packageList.forEach((value) => print("External library: $value"));
+    });
+    ```
+
+6. Enable log callback.
+    ```
+    void logCallback(int level, String message) {
+        print(message);
+    }
+    ...
+    _flutterFFmpeg.enableLogCallback(this.logCallback);
+    ```
+
+7. Enable statistics callback.
+    ```
+    void statisticsCallback(int time, int size, double bitrate, double speed, int videoFrameNumber, double videoQuality, double videoFps) {
+        print("Statistics: time: $time, size: $size, bitrate: $bitrate, speed: $speed, videoFrameNumber: $videoFrameNumber, videoQuality: $videoQuality, videoFps: $videoFps");
+    }
+    ...
+    _flutterFFmpeg.enableStatisticsCallback(this.statisticsCallback);
+    ```
+
+8. Get last received statistics.
+    ```
+    _flutterFFmpeg.getLastReceivedStatistics().then((stats) => print(stats));
+    ```
+
+9. Set log level.
+    ```
+    _flutterFFmpeg.setLogLevel(LogLevel.AV_LOG_WARNING);
+    ```
+
+10. Register custom fonts directory.
+    ```
+    _flutterFFmpeg.setFontDirectory("<folder with fonts>");
+    ```
+
+### 4. Versions
+
+#### 4.1 Releases
+
+- `0.1.0` releases is based on `FFmpeg v4.2-dev` and `MobileFFmpeg v4.2.LTS`
+
+### 5. Updates
+
+Refer to [Changelog](CHANGELOG.md) for updates.
+
+### 6. License
+
+This project is licensed under the LGPL v3.0. However, if installation is customized to use a package with `-gpl` postfix (min-gpl, https-gpl, full-gpl) then `FlutterFFmpeg` is subject to the GPL v3.0 license.
+
+Digital assets used in test applications are published in the public domain.
+
+### 7. Contributing
+
+Feel free to submit issues or pull requests.
+
+### 8. See Also
+
+- [FFmpeg](https://www.ffmpeg.org)
+- [Mobile FFmpeg Wiki](https://github.com/tanersener/mobile-ffmpeg/wiki)
+- [FFmpeg License and Legal Considerations](https://ffmpeg.org/legal.html)

+ 0 - 11
example/README.md

@@ -3,14 +3,3 @@
 Demonstrates how to use the flutter_ffmpeg plugin.
 
 ## Getting Started
-
-This project is a starting point for a Flutter application.
-
-A few resources to get you started if this is your first Flutter project:
-
-- [Lab: Write your first Flutter app](https://flutter.io/docs/get-started/codelab)
-- [Cookbook: Useful Flutter samples](https://flutter.io/docs/cookbook)
-
-For help getting started with Flutter, view our 
-[online documentation](https://flutter.io/docs), which offers tutorials, 
-samples, guidance on mobile development, and a full API reference.

+ 1 - 1
example/pubspec.yaml

@@ -3,7 +3,7 @@ description: Demonstrates how to use the flutter_ffmpeg plugin.
 publish_to: 'none'
 
 environment:
-  sdk: ">=2.0.0-dev.68.0 <3.0.0"
+  sdk: ">=2.0.0-dev.68.0"
 
 dependencies:
   flutter:

+ 2 - 0
flutter_ffmpeg.iml

@@ -11,7 +11,9 @@
       <excludeFolder url="file://$MODULE_DIR$/example/.dart_tool" />
       <excludeFolder url="file://$MODULE_DIR$/example/.pub" />
       <excludeFolder url="file://$MODULE_DIR$/example/build" />
+      <excludeFolder url="file://$MODULE_DIR$/example/ios/.symlinks/plugins/flutter_ffmpeg/packages" />
       <excludeFolder url="file://$MODULE_DIR$/example/ios/Flutter/flutter_assets/packages" />
+      <excludeFolder url="file://$MODULE_DIR$/packages" />
     </content>
     <orderEntry type="sourceFolder" forTests="false" />
     <orderEntry type="library" name="Dart SDK" level="project" />

+ 7 - 0
packages/flutter_ffmpeg_audio/.gitignore

@@ -0,0 +1,7 @@
+.idea/
+.DS_Store
+.dart_tool/
+.pub/
+build/
+pubspec.lock
+.packages

+ 10 - 0
packages/flutter_ffmpeg_audio/.metadata

@@ -0,0 +1,10 @@
+# This file tracks properties of this Flutter project.
+# Used by Flutter tool to assess capabilities and perform upgrades etc.
+#
+# This file should be version controlled and should not be manually edited.
+
+version:
+  revision: 5391447fae6209bb21a89e6a5a6583cac1af9b4b
+  channel: stable
+
+project_type: plugin

+ 165 - 0
packages/flutter_ffmpeg_audio/LICENSE

@@ -0,0 +1,165 @@
+                   GNU LESSER GENERAL PUBLIC LICENSE
+                       Version 3, 29 June 2007
+
+ Copyright (C) 2007 Free Software Foundation, Inc. <http://fsf.org/>
+ Everyone is permitted to copy and distribute verbatim copies
+ of this license document, but changing it is not allowed.
+
+
+  This version of the GNU Lesser General Public License incorporates
+the terms and conditions of version 3 of the GNU General Public
+License, supplemented by the additional permissions listed below.
+
+  0. Additional Definitions.
+
+  As used herein, "this License" refers to version 3 of the GNU Lesser
+General Public License, and the "GNU GPL" refers to version 3 of the GNU
+General Public License.
+
+  "The Library" refers to a covered work governed by this License,
+other than an Application or a Combined Work as defined below.
+
+  An "Application" is any work that makes use of an interface provided
+by the Library, but which is not otherwise based on the Library.
+Defining a subclass of a class defined by the Library is deemed a mode
+of using an interface provided by the Library.
+
+  A "Combined Work" is a work produced by combining or linking an
+Application with the Library.  The particular version of the Library
+with which the Combined Work was made is also called the "Linked
+Version".
+
+  The "Minimal Corresponding Source" for a Combined Work means the
+Corresponding Source for the Combined Work, excluding any source code
+for portions of the Combined Work that, considered in isolation, are
+based on the Application, and not on the Linked Version.
+
+  The "Corresponding Application Code" for a Combined Work means the
+object code and/or source code for the Application, including any data
+and utility programs needed for reproducing the Combined Work from the
+Application, but excluding the System Libraries of the Combined Work.
+
+  1. Exception to Section 3 of the GNU GPL.
+
+  You may convey a covered work under sections 3 and 4 of this License
+without being bound by section 3 of the GNU GPL.
+
+  2. Conveying Modified Versions.
+
+  If you modify a copy of the Library, and, in your modifications, a
+facility refers to a function or data to be supplied by an Application
+that uses the facility (other than as an argument passed when the
+facility is invoked), then you may convey a copy of the modified
+version:
+
+   a) under this License, provided that you make a good faith effort to
+   ensure that, in the event an Application does not supply the
+   function or data, the facility still operates, and performs
+   whatever part of its purpose remains meaningful, or
+
+   b) under the GNU GPL, with none of the additional permissions of
+   this License applicable to that copy.
+
+  3. Object Code Incorporating Material from Library Header Files.
+
+  The object code form of an Application may incorporate material from
+a header file that is part of the Library.  You may convey such object
+code under terms of your choice, provided that, if the incorporated
+material is not limited to numerical parameters, data structure
+layouts and accessors, or small macros, inline functions and templates
+(ten or fewer lines in length), you do both of the following:
+
+   a) Give prominent notice with each copy of the object code that the
+   Library is used in it and that the Library and its use are
+   covered by this License.
+
+   b) Accompany the object code with a copy of the GNU GPL and this license
+   document.
+
+  4. Combined Works.
+
+  You may convey a Combined Work under terms of your choice that,
+taken together, effectively do not restrict modification of the
+portions of the Library contained in the Combined Work and reverse
+engineering for debugging such modifications, if you also do each of
+the following:
+
+   a) Give prominent notice with each copy of the Combined Work that
+   the Library is used in it and that the Library and its use are
+   covered by this License.
+
+   b) Accompany the Combined Work with a copy of the GNU GPL and this license
+   document.
+
+   c) For a Combined Work that displays copyright notices during
+   execution, include the copyright notice for the Library among
+   these notices, as well as a reference directing the user to the
+   copies of the GNU GPL and this license document.
+
+   d) Do one of the following:
+
+       0) Convey the Minimal Corresponding Source under the terms of this
+       License, and the Corresponding Application Code in a form
+       suitable for, and under terms that permit, the user to
+       recombine or relink the Application with a modified version of
+       the Linked Version to produce a modified Combined Work, in the
+       manner specified by section 6 of the GNU GPL for conveying
+       Corresponding Source.
+
+       1) Use a suitable shared library mechanism for linking with the
+       Library.  A suitable mechanism is one that (a) uses at run time
+       a copy of the Library already present on the user's computer
+       system, and (b) will operate properly with a modified version
+       of the Library that is interface-compatible with the Linked
+       Version.
+
+   e) Provide Installation Information, but only if you would otherwise
+   be required to provide such information under section 6 of the
+   GNU GPL, and only to the extent that such information is
+   necessary to install and execute a modified version of the
+   Combined Work produced by recombining or relinking the
+   Application with a modified version of the Linked Version. (If
+   you use option 4d0, the Installation Information must accompany
+   the Minimal Corresponding Source and Corresponding Application
+   Code. If you use option 4d1, you must provide the Installation
+   Information in the manner specified by section 6 of the GNU GPL
+   for conveying Corresponding Source.)
+
+  5. Combined Libraries.
+
+  You may place library facilities that are a work based on the
+Library side by side in a single library together with other library
+facilities that are not Applications and are not covered by this
+License, and convey such a combined library under terms of your
+choice, if you do both of the following:
+
+   a) Accompany the combined library with a copy of the same work based
+   on the Library, uncombined with any other library facilities,
+   conveyed under the terms of this License.
+
+   b) Give prominent notice with the combined library that part of it
+   is a work based on the Library, and explaining where to find the
+   accompanying uncombined form of the same work.
+
+  6. Revised Versions of the GNU Lesser General Public License.
+
+  The Free Software Foundation may publish revised and/or new versions
+of the GNU Lesser General Public License from time to time. Such new
+versions will be similar in spirit to the present version, but may
+differ in detail to address new problems or concerns.
+
+  Each version is given a distinguishing version number. If the
+Library as you received it specifies that a certain numbered version
+of the GNU Lesser General Public License "or any later version"
+applies to it, you have the option of following the terms and
+conditions either of that published version or of any later version
+published by the Free Software Foundation. If the Library as you
+received it does not specify a version number of the GNU Lesser
+General Public License, you may choose any version of the GNU Lesser
+General Public License ever published by the Free Software Foundation.
+
+  If the Library as you received it specifies that a proxy can decide
+whether future versions of the GNU Lesser General Public License shall
+apply, that proxy's public statement of acceptance of any version is
+permanent authorization for you to choose that version for the
+Library.

+ 8 - 0
packages/flutter_ffmpeg_audio/android/.gitignore

@@ -0,0 +1,8 @@
+*.iml
+.gradle
+/local.properties
+/.idea/workspace.xml
+/.idea/libraries
+.DS_Store
+/build
+/captures

+ 38 - 0
packages/flutter_ffmpeg_audio/android/build.gradle

@@ -0,0 +1,38 @@
+group 'com.arthenica.flutter.ffmpeg'
+version '0.1.0'
+
+buildscript {
+    repositories {
+        google()
+        jcenter()
+    }
+
+    dependencies {
+        classpath 'com.android.tools.build:gradle:3.2.1'
+    }
+}
+
+rootProject.allprojects {
+    repositories {
+        google()
+        jcenter()
+    }
+}
+
+apply plugin: 'com.android.library'
+
+android {
+    compileSdkVersion 27
+
+    defaultConfig {
+        minSdkVersion 21
+        testInstrumentationRunner "android.support.test.runner.AndroidJUnitRunner"
+    }
+    lintOptions {
+        disable 'InvalidPackage'
+    }
+}
+
+dependencies {
+    implementation 'com.arthenica:mobile-ffmpeg-audio:4.2.LTS'
+}

+ 1 - 0
packages/flutter_ffmpeg_audio/android/gradle.properties

@@ -0,0 +1 @@
+org.gradle.jvmargs=-Xmx1536M

+ 1 - 0
packages/flutter_ffmpeg_audio/android/settings.gradle

@@ -0,0 +1 @@
+rootProject.name = 'flutter_ffmpeg'

+ 3 - 0
packages/flutter_ffmpeg_audio/android/src/main/AndroidManifest.xml

@@ -0,0 +1,3 @@
+<manifest xmlns:android="http://schemas.android.com/apk/res/android"
+  package="com.arthenica.flutter.ffmpeg">
+</manifest>

+ 66 - 0
packages/flutter_ffmpeg_audio/android/src/main/java/com/arthenica/flutter/ffmpeg/FlutterFFmpegExecuteAsyncArgumentsTask.java

@@ -0,0 +1,66 @@
+/*
+ * Copyright (c) 2019 Taner Sener
+ *
+ * This file is part of FlutterFFmpeg.
+ *
+ * FlutterFFmpeg is free software: you can redistribute it and/or modify
+ * it under the terms of the GNU Lesser General Public License as published by
+ * the Free Software Foundation, either version 3 of the License, or
+ * (at your option) any later version.
+ *
+ * FlutterFFmpeg is distributed in the hope that it will be useful,
+ * but WITHOUT ANY WARRANTY; without even the implied warranty of
+ * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
+ * GNU Lesser General Public License for more details.
+ *
+ * You should have received a copy of the GNU Lesser General Public License
+ * along with FlutterFFmpeg.  If not, see <http://www.gnu.org/licenses/>.
+ */
+
+package com.arthenica.flutter.ffmpeg;
+
+import android.os.AsyncTask;
+import android.util.Log;
+
+import com.arthenica.mobileffmpeg.FFmpeg;
+
+import java.util.Arrays;
+import java.util.List;
+
+import io.flutter.plugin.common.MethodChannel;
+
+/**
+ * Asynchronous task which performs {@link FFmpeg#execute(String[])} method invocations.
+ *
+ * @author Taner Sener
+ * @since 0.1.0
+ */
+public class FlutterFFmpegExecuteAsyncArgumentsTask extends AsyncTask<String, Integer, Integer> {
+
+    private final MethodChannel.Result result;
+    private final List<String> arguments;
+
+    FlutterFFmpegExecuteAsyncArgumentsTask(final List<String> arguments, final MethodChannel.Result result) {
+        this.arguments = arguments;
+        this.result = result;
+    }
+
+    @Override
+    protected Integer doInBackground(final String... dummyString) {
+        final String[] argumentsArray = arguments.toArray(new String[0]);
+
+        Log.d(FlutterFFmpegPlugin.LIBRARY_NAME, String.format("Running FFmpeg with arguments: %s.", Arrays.toString(argumentsArray)));
+
+        int rc = FFmpeg.execute(argumentsArray);
+
+        Log.d(FlutterFFmpegPlugin.LIBRARY_NAME, String.format("FFmpeg exited with rc: %d", rc));
+
+        return rc;
+    }
+
+    @Override
+    protected void onPostExecute(final Integer rc) {
+        result.success(FlutterFFmpegPlugin.toIntMap(FlutterFFmpegPlugin.KEY_RC, rc));
+    }
+
+}

+ 72 - 0
packages/flutter_ffmpeg_audio/android/src/main/java/com/arthenica/flutter/ffmpeg/FlutterFFmpegExecuteAsyncCommandTask.java

@@ -0,0 +1,72 @@
+/*
+ * Copyright (c) 2019 Taner Sener
+ *
+ * This file is part of FlutterFFmpeg.
+ *
+ * FlutterFFmpeg is free software: you can redistribute it and/or modify
+ * it under the terms of the GNU Lesser General Public License as published by
+ * the Free Software Foundation, either version 3 of the License, or
+ * (at your option) any later version.
+ *
+ * FlutterFFmpeg is distributed in the hope that it will be useful,
+ * but WITHOUT ANY WARRANTY; without even the implied warranty of
+ * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
+ * GNU Lesser General Public License for more details.
+ *
+ * You should have received a copy of the GNU Lesser General Public License
+ * along with FlutterFFmpeg.  If not, see <http://www.gnu.org/licenses/>.
+ */
+
+package com.arthenica.flutter.ffmpeg;
+
+import android.os.AsyncTask;
+import android.util.Log;
+
+import com.arthenica.mobileffmpeg.FFmpeg;
+
+import io.flutter.plugin.common.MethodChannel;
+
+/**
+ * Asynchronous task which performs {@link FFmpeg#execute(String, String)} method invocations.
+ *
+ * @author Taner Sener
+ * @since 0.1.0
+ */
+public class FlutterFFmpegExecuteAsyncCommandTask extends AsyncTask<String, Integer, Integer> {
+
+    private String delimiter;
+    private final MethodChannel.Result result;
+
+    FlutterFFmpegExecuteAsyncCommandTask(final String delimiter, final MethodChannel.Result result) {
+        if (delimiter == null) {
+            this.delimiter = " ";
+        } else {
+            this.delimiter = delimiter;
+        }
+
+        this.result = result;
+    }
+
+    @Override
+    protected Integer doInBackground(final String... strings) {
+        int rc = -1;
+
+        if ((strings != null) && (strings.length > 0)) {
+            final String command = strings[0];
+
+            Log.d(FlutterFFmpegPlugin.LIBRARY_NAME, String.format("Running FFmpeg command: %s with delimiter %s.", command, delimiter));
+
+            rc = FFmpeg.execute(command, delimiter);
+
+            Log.d(FlutterFFmpegPlugin.LIBRARY_NAME, String.format("FFmpeg exited with rc: %d", rc));
+        }
+
+        return rc;
+    }
+
+    @Override
+    protected void onPostExecute(final Integer rc) {
+        result.success(FlutterFFmpegPlugin.toIntMap(FlutterFFmpegPlugin.KEY_RC, rc));
+    }
+
+}

+ 70 - 0
packages/flutter_ffmpeg_audio/android/src/main/java/com/arthenica/flutter/ffmpeg/FlutterFFmpegGetMediaInformationAsyncTask.java

@@ -0,0 +1,70 @@
+/*
+ * Copyright (c) 2019 Taner Sener
+ *
+ * This file is part of FlutterFFmpeg.
+ *
+ * FlutterFFmpeg is free software: you can redistribute it and/or modify
+ * it under the terms of the GNU Lesser General Public License as published by
+ * the Free Software Foundation, either version 3 of the License, or
+ * (at your option) any later version.
+ *
+ * FlutterFFmpeg is distributed in the hope that it will be useful,
+ * but WITHOUT ANY WARRANTY; without even the implied warranty of
+ * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
+ * GNU Lesser General Public License for more details.
+ *
+ * You should have received a copy of the GNU Lesser General Public License
+ * along with FlutterFFmpeg.  If not, see <http://www.gnu.org/licenses/>.
+ */
+
+package com.arthenica.flutter.ffmpeg;
+
+import android.os.AsyncTask;
+import android.util.Log;
+
+import com.arthenica.mobileffmpeg.FFmpeg;
+import com.arthenica.mobileffmpeg.MediaInformation;
+
+import io.flutter.plugin.common.MethodChannel;
+
+/**
+ * Asynchronous task which performs {@link FFmpeg#getMediaInformation(String, Long)} method invocations.
+ *
+ * @author Taner Sener
+ * @since 0.1.0
+ */
+public class FlutterFFmpegGetMediaInformationAsyncTask extends AsyncTask<String, Integer, MediaInformation> {
+
+    private Integer timeout;
+    private final MethodChannel.Result result;
+
+    FlutterFFmpegGetMediaInformationAsyncTask(final Integer timeout, final MethodChannel.Result result) {
+        this.timeout = timeout;
+        this.result = result;
+    }
+
+    @Override
+    protected MediaInformation doInBackground(final String... strings) {
+        MediaInformation mediaInformation = null;
+
+        if ((strings != null) && (strings.length > 0)) {
+            final String path = strings[0];
+
+            if (timeout == null) {
+                Log.d(FlutterFFmpegPlugin.LIBRARY_NAME, String.format("Getting media information for %s", path));
+                mediaInformation = FFmpeg.getMediaInformation(path);
+            } else {
+                Log.d(FlutterFFmpegPlugin.LIBRARY_NAME, String.format("Getting media information for %s with timeout %d.", path, timeout.longValue()));
+                mediaInformation = FFmpeg.getMediaInformation(path, timeout.longValue());
+            }
+        }
+
+        return mediaInformation;
+    }
+
+    @Override
+    protected void onPostExecute(final MediaInformation mediaInformation) {
+        result.success(FlutterFFmpegPlugin.toMediaInformationMap(mediaInformation));
+    }
+
+}

+ 436 - 0
packages/flutter_ffmpeg_audio/android/src/main/java/com/arthenica/flutter/ffmpeg/FlutterFFmpegPlugin.java

@@ -0,0 +1,436 @@
+/*
+ * Copyright (c) 2019 Taner Sener
+ *
+ * This file is part of FlutterFFmpeg.
+ *
+ * FlutterFFmpeg is free software: you can redistribute it and/or modify
+ * it under the terms of the GNU Lesser General Public License as published by
+ * the Free Software Foundation, either version 3 of the License, or
+ * (at your option) any later version.
+ *
+ * FlutterFFmpeg is distributed in the hope that it will be useful,
+ * but WITHOUT ANY WARRANTY; without even the implied warranty of
+ * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
+ * GNU Lesser General Public License for more details.
+ *
+ * You should have received a copy of the GNU Lesser General Public License
+ * along with FlutterFFmpeg.  If not, see <http://www.gnu.org/licenses/>.
+ */
+
+package com.arthenica.flutter.ffmpeg;
+
+import android.content.Context;
+import android.system.ErrnoException;
+import android.util.Log;
+
+import com.arthenica.mobileffmpeg.AbiDetect;
+import com.arthenica.mobileffmpeg.Config;
+import com.arthenica.mobileffmpeg.FFmpeg;
+import com.arthenica.mobileffmpeg.Level;
+import com.arthenica.mobileffmpeg.LogCallback;
+import com.arthenica.mobileffmpeg.LogMessage;
+import com.arthenica.mobileffmpeg.MediaInformation;
+import com.arthenica.mobileffmpeg.Statistics;
+import com.arthenica.mobileffmpeg.StatisticsCallback;
+import com.arthenica.mobileffmpeg.StreamInformation;
+
+import java.util.ArrayList;
+import java.util.HashMap;
+import java.util.List;
+import java.util.Map;
+import java.util.Set;
+
+import io.flutter.plugin.common.EventChannel;
+import io.flutter.plugin.common.MethodCall;
+import io.flutter.plugin.common.MethodChannel;
+import io.flutter.plugin.common.MethodChannel.MethodCallHandler;
+import io.flutter.plugin.common.MethodChannel.Result;
+import io.flutter.plugin.common.PluginRegistry.Registrar;
+
+/**
+ * <h3>Flutter FFmpeg Plugin</h3>
+ *
+ * @author Taner Sener
+ * @since 0.1.0
+ */
+public class FlutterFFmpegPlugin implements MethodCallHandler, EventChannel.StreamHandler {
+    public static final String LIBRARY_NAME = "flutter-ffmpeg";
+
+    public static final String PLATFORM_NAME = "android";
+    public static final String KEY_VERSION = "version";
+    public static final String KEY_RC = "rc";
+    public static final String KEY_PLATFORM = "platform";
+    public static final String KEY_PACKAGE_NAME = "packageName";
+    public static final String KEY_LAST_RC = "lastRc";
+
+    public static final String KEY_LAST_COMMAND_OUTPUT = "lastCommandOutput";
+    public static final String KEY_LOG_TEXT = "log";
+
+    public static final String KEY_LOG_LEVEL = "level";
+    public static final String KEY_STAT_TIME = "time";
+    public static final String KEY_STAT_SIZE = "size";
+    public static final String KEY_STAT_BITRATE = "bitrate";
+    public static final String KEY_STAT_SPEED = "speed";
+    public static final String KEY_STAT_VIDEO_FRAME_NUMBER = "videoFrameNumber";
+    public static final String KEY_STAT_VIDEO_QUALITY = "videoQuality";
+    public static final String KEY_STAT_VIDEO_FPS = "videoFps";
+
+    public static final String EVENT_LOG = "FlutterFFmpegLogCallback";
+    public static final String EVENT_STAT = "FlutterFFmpegStatisticsCallback";
+
+    private EventChannel.EventSink eventSink;
+    private final Registrar registrar;
+
+    /**
+     * Registers plugin to registry.
+     *
+     * @param registrar receiver of plugin registration
+     */
+    public static void registerWith(final Registrar registrar) {
+        FlutterFFmpegPlugin handler = new FlutterFFmpegPlugin(registrar);
+
+        final MethodChannel channel = new MethodChannel(registrar.messenger(), "flutter_ffmpeg");
+        channel.setMethodCallHandler(handler);
+
+        final EventChannel eventChannel = new EventChannel(registrar.messenger(), "flutter_ffmpeg_event");
+        eventChannel.setStreamHandler(handler);
+    }
+
+    private FlutterFFmpegPlugin(Registrar registrar) {
+        this.registrar = registrar;
+    }
+
+    private Context getActiveContext() {
+        return (registrar.activity() != null) ? registrar.activity() : registrar.context();
+    }
+
+    /**
+     * Handles method calls.
+     *
+     * @param call   method call
+     * @param result result callback
+     */
+    @Override
+    public void onMethodCall(final MethodCall call, final Result result) {
+        if (call.method.equals("getPlatform")) {
+
+            final String abi = AbiDetect.getAbi();
+            result.success(toStringMap(KEY_PLATFORM, PLATFORM_NAME + "-" + abi));
+
+        } else if (call.method.equals("getFFmpegVersion")) {
+
+            final String version = FFmpeg.getFFmpegVersion();
+            result.success(toStringMap(KEY_VERSION, version));
+
+        } else if (call.method.equals("executeWithArguments")) {
+
+            List<String> arguments = call.argument("arguments");
+
+            final FlutterFFmpegExecuteAsyncArgumentsTask asyncTask = new FlutterFFmpegExecuteAsyncArgumentsTask(arguments, result);
+            asyncTask.execute("dummy-trigger");
+
+        } else if (call.method.equals("execute")) {
+
+            String command = call.argument("command");
+            String delimiter = call.argument("delimiter");
+
+            final FlutterFFmpegExecuteAsyncCommandTask asyncTask = new FlutterFFmpegExecuteAsyncCommandTask(delimiter, result);
+            asyncTask.execute(command);
+
+        } else if (call.method.equals("cancel")) {
+
+            FFmpeg.cancel();
+
+        } else if (call.method.equals("enableRedirection")) {
+
+            Config.enableRedirection();
+
+        } else if (call.method.equals("disableRedirection")) {
+
+            Config.disableRedirection();
+
+        } else if (call.method.equals("getLogLevel")) {
+
+            final Level level = Config.getLogLevel();
+            result.success(toIntMap(KEY_LOG_LEVEL, levelToInt(level)));
+
+        } else if (call.method.equals("setLogLevel")) {
+
+            Integer level = call.argument("level");
+            if (level == null) {
+                level = Level.AV_LOG_TRACE.getValue();
+            }
+            Config.setLogLevel(Level.from(level));
+
+        } else if (call.method.equals("enableLogs")) {
+
+            Config.enableLogCallback(new LogCallback() {
+
+                @Override
+                public void apply(final LogMessage logMessage) {
+                    emitLogMessage(logMessage);
+                }
+            });
+
+        } else if (call.method.equals("disableLogs")) {
+
+            Config.enableLogCallback(null);
+
+        } else if (call.method.equals("enableStatistics")) {
+
+            Config.enableStatisticsCallback(new StatisticsCallback() {
+
+                @Override
+                public void apply(final Statistics statistics) {
+                    emitStatistics(statistics);
+                }
+            });
+
+        } else if (call.method.equals("disableStatistics")) {
+
+            Config.enableStatisticsCallback(null);
+
+        } else if (call.method.equals("getLastReceivedStatistics")) {
+
+            result.success(toMap(Config.getLastReceivedStatistics()));
+
+        } else if (call.method.equals("resetStatistics")) {
+
+            Config.resetStatistics();
+
+        } else if (call.method.equals("setFontconfigConfigurationPath")) {
+            String path = call.argument("path");
+
+            try {
+                Config.setFontconfigConfigurationPath(path);
+            } catch (final ErrnoException e) {
+                Log.w(LIBRARY_NAME, String.format("Setting fontconfig configuration path failed for %s", path), e);
+            }
+
+        } else if (call.method.equals("setFontDirectory")) {
+
+            String path = call.argument("fontDirectory");
+            Map<String, String> map = call.argument("fontNameMap");
+
+            Config.setFontDirectory(getActiveContext(), path, map);
+
+        } else if (call.method.equals("getPackageName")) {
+
+            final String packageName = Config.getPackageName();
+            result.success(toStringMap(KEY_PACKAGE_NAME, packageName));
+
+        } else if (call.method.equals("getExternalLibraries")) {
+
+            final List<String> externalLibraries = Config.getExternalLibraries();
+            result.success(externalLibraries);
+
+        } else if (call.method.equals("getLastReturnCode")) {
+
+            int lastReturnCode = FFmpeg.getLastReturnCode();
+            result.success(toIntMap(KEY_LAST_RC, lastReturnCode));
+
+        } else if (call.method.equals("getLastCommandOutput")) {
+
+            final String lastCommandOutput = FFmpeg.getLastCommandOutput();
+            result.success(toStringMap(KEY_LAST_COMMAND_OUTPUT, lastCommandOutput));
+
+        } else if (call.method.equals("getMediaInformation")) {
+            final String path = call.argument("path");
+            Integer timeout = call.argument("timeout");
+            if (timeout == null) {
+                timeout = 10000;
+            }
+
+            final FlutterFFmpegGetMediaInformationAsyncTask asyncTask = new FlutterFFmpegGetMediaInformationAsyncTask(timeout, result);
+            asyncTask.execute(path);
+
+        } else {
+            result.notImplemented();
+        }
+    }
+
+    @Override
+    public void onListen(Object o, EventChannel.EventSink eventSink) {
+        this.eventSink = eventSink;
+    }
+
+    @Override
+    public void onCancel(Object o) {
+        this.eventSink = null;
+    }
+
+    protected void emitLogMessage(final LogMessage logMessage) {
+        final HashMap<String, Object> logWrapperMap = new HashMap<>();
+        final HashMap<String, Object> logMap = new HashMap<>();
+
+        logMap.put(KEY_LOG_LEVEL, levelToInt(logMessage.getLevel()));
+        logMap.put(KEY_LOG_TEXT, logMessage.getText());
+
+        logWrapperMap.put(EVENT_LOG, logMap);
+
+        eventSink.success(logWrapperMap);
+    }
+
+    protected void emitStatistics(final Statistics statistics) {
+        final HashMap<String, Object> statisticsMap = new HashMap<>();
+        statisticsMap.put(EVENT_STAT, toMap(statistics));
+        eventSink.success(statisticsMap);
+    }
+
+    public static int levelToInt(final Level level) {
+        return (level == null) ? Level.AV_LOG_TRACE.getValue() : level.getValue();
+    }
+
+    public static HashMap<String, String> toStringMap(final String key, final String value) {
+        final HashMap<String, String> map = new HashMap<>();
+        map.put(key, value);
+        return map;
+    }
+
+    public static HashMap<String, Integer> toIntMap(final String key, final int value) {
+        final HashMap<String, Integer> map = new HashMap<>();
+        map.put(key, value);
+        return map;
+    }
+
+    public static Map<String, Object> toMap(final Statistics statistics) {
+        final HashMap<String, Object> statisticsMap = new HashMap<>();
+
+        if (statistics != null) {
+            statisticsMap.put(KEY_STAT_TIME, statistics.getTime());
+            statisticsMap.put(KEY_STAT_SIZE, (statistics.getSize() < Integer.MAX_VALUE) ? (int) statistics.getSize() : (int) (statistics.getSize() % Integer.MAX_VALUE));
+            statisticsMap.put(KEY_STAT_BITRATE, statistics.getBitrate());
+            statisticsMap.put(KEY_STAT_SPEED, statistics.getSpeed());
+
+            statisticsMap.put(KEY_STAT_VIDEO_FRAME_NUMBER, statistics.getVideoFrameNumber());
+            statisticsMap.put(KEY_STAT_VIDEO_QUALITY, statistics.getVideoQuality());
+            statisticsMap.put(KEY_STAT_VIDEO_FPS, statistics.getVideoFps());
+        }
+
+        return statisticsMap;
+    }
+
+    public static HashMap<String, Object> toMediaInformationMap(final MediaInformation mediaInformation) {
+        final HashMap<String, Object> map = new HashMap<>();
+
+        if (mediaInformation != null) {
+            if (mediaInformation.getFormat() != null) {
+                map.put("format", mediaInformation.getFormat());
+            }
+            if (mediaInformation.getPath() != null) {
+                map.put("path", mediaInformation.getPath());
+            }
+            if (mediaInformation.getStartTime() != null) {
+                map.put("startTime", mediaInformation.getStartTime().intValue());
+            }
+            if (mediaInformation.getDuration() != null) {
+                map.put("duration", mediaInformation.getDuration().intValue());
+            }
+            if (mediaInformation.getBitrate() != null) {
+                map.put("bitrate", mediaInformation.getBitrate().intValue());
+            }
+            if (mediaInformation.getRawInformation() != null) {
+                map.put("rawInformation", mediaInformation.getRawInformation());
+            }
+
+            final Set<Map.Entry<String, String>> metadata = mediaInformation.getMetadataEntries();
+            if ((metadata != null) && (metadata.size() > 0)) {
+                final HashMap<String, String> metadataMap = new HashMap<>();
+
+                for (Map.Entry<String, String> entry : metadata) {
+                    metadataMap.put(entry.getKey(), entry.getValue());
+                }
+
+                map.put("metadata", metadataMap);
+            }
+
+            final List<StreamInformation> streams = mediaInformation.getStreams();
+            if ((streams != null) && (streams.size() > 0)) {
+                final ArrayList<Map<String, Object>> array = new ArrayList<>();
+
+                for (StreamInformation streamInformation : streams) {
+                    array.add(toStreamInformationMap(streamInformation));
+                }
+
+                map.put("streams", array);
+            }
+        }
+
+        return map;
+    }
+
+    public static Map<String, Object> toStreamInformationMap(final StreamInformation streamInformation) {
+        final HashMap<String, Object> map = new HashMap<>();
+
+        if (streamInformation != null) {
+            if (streamInformation.getIndex() != null) {
+                map.put("index", streamInformation.getIndex().intValue());
+            }
+            if (streamInformation.getType() != null) {
+                map.put("type", streamInformation.getType());
+            }
+            if (streamInformation.getCodec() != null) {
+                map.put("codec", streamInformation.getCodec());
+            }
+            if (streamInformation.getFullCodec() != null) {
+                map.put("fullCodec", streamInformation.getFullCodec());
+            }
+            if (streamInformation.getFormat() != null) {
+                map.put("format", streamInformation.getFormat());
+            }
+            if (streamInformation.getFullFormat() != null) {
+                map.put("fullFormat", streamInformation.getFullFormat());
+            }
+            if (streamInformation.getWidth() != null) {
+                map.put("width", streamInformation.getWidth().intValue());
+            }
+            if (streamInformation.getHeight() != null) {
+                map.put("height", streamInformation.getHeight().intValue());
+            }
+            if (streamInformation.getBitrate() != null) {
+                map.put("bitrate", streamInformation.getBitrate().intValue());
+            }
+            if (streamInformation.getSampleRate() != null) {
+                map.put("sampleRate", streamInformation.getSampleRate().intValue());
+            }
+            if (streamInformation.getSampleFormat() != null) {
+                map.put("sampleFormat", streamInformation.getSampleFormat());
+            }
+            if (streamInformation.getChannelLayout() != null) {
+                map.put("channelLayout", streamInformation.getChannelLayout());
+            }
+            if (streamInformation.getSampleAspectRatio() != null) {
+                map.put("sampleAspectRatio", streamInformation.getSampleAspectRatio());
+            }
+            if (streamInformation.getDisplayAspectRatio() != null) {
+                map.put("displayAspectRatio", streamInformation.getDisplayAspectRatio());
+            }
+            if (streamInformation.getAverageFrameRate() != null) {
+                map.put("averageFrameRate", streamInformation.getAverageFrameRate());
+            }
+            if (streamInformation.getRealFrameRate() != null) {
+                map.put("realFrameRate", streamInformation.getRealFrameRate());
+            }
+            if (streamInformation.getTimeBase() != null) {
+                map.put("timeBase", streamInformation.getTimeBase());
+            }
+            if (streamInformation.getCodecTimeBase() != null) {
+                map.put("codecTimeBase", streamInformation.getCodecTimeBase());
+            }
+
+            final Set<Map.Entry<String, String>> metadata = streamInformation.getMetadataEntries();
+            if ((metadata != null) && (metadata.size() > 0)) {
+                final HashMap<String, String> metadataMap = new HashMap<>();
+
+                for (Map.Entry<String, String> entry : metadata) {
+                    metadataMap.put(entry.getKey(), entry.getValue());
+                }
+
+                map.put("metadata", metadataMap);
+            }
+        }
+
+        return map;
+    }
+
+}

+ 36 - 0
packages/flutter_ffmpeg_audio/ios/.gitignore

@@ -0,0 +1,36 @@
+.idea/
+.vagrant/
+.sconsign.dblite
+.svn/
+
+.DS_Store
+*.swp
+profile
+
+DerivedData/
+build/
+GeneratedPluginRegistrant.h
+GeneratedPluginRegistrant.m
+
+.generated/
+
+*.pbxuser
+*.mode1v3
+*.mode2v3
+*.perspectivev3
+
+!default.pbxuser
+!default.mode1v3
+!default.mode2v3
+!default.perspectivev3
+
+xcuserdata
+
+*.moved-aside
+
+*.pyc
+*sync/
+Icon?
+.tags*
+
+/Flutter/Generated.xcconfig

+ 0 - 0
packages/flutter_ffmpeg_audio/ios/Assets/.gitkeep


+ 27 - 0
packages/flutter_ffmpeg_audio/ios/Classes/FlutterFFmpegPlugin.h

@@ -0,0 +1,27 @@
+/*
+ * Copyright (c) 2019 Taner Sener
+ *
+ * This file is part of FlutterFFmpeg.
+ *
+ * FlutterFFmpeg is free software: you can redistribute it and/or modify
+ * it under the terms of the GNU Lesser General Public License as published by
+ * the Free Software Foundation, either version 3 of the License, or
+ * (at your option) any later version.
+ *
+ * FlutterFFmpeg is distributed in the hope that it will be useful,
+ * but WITHOUT ANY WARRANTY; without even the implied warranty of
+ * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
+ * GNU Lesser General Public License for more details.
+ *
+ * You should have received a copy of the GNU Lesser General Public License
+ * along with FlutterFFmpeg.  If not, see <http://www.gnu.org/licenses/>.
+ */
+
+#import <Flutter/Flutter.h>
+#import <mobileffmpeg/MobileFFmpegConfig.h>
+
+/**
+ * Flutter FFmpeg Plugin
+ */
+@interface FlutterFFmpegPlugin : NSObject<FlutterPlugin,FlutterStreamHandler,LogDelegate,StatisticsDelegate>
+@end

+ 387 - 0
packages/flutter_ffmpeg_audio/ios/Classes/FlutterFfmpegPlugin.m

@@ -0,0 +1,387 @@
+/*
+ * Copyright (c) 2019 Taner Sener
+ *
+ * This file is part of FlutterFFmpeg.
+ *
+ * FlutterFFmpeg is free software: you can redistribute it and/or modify
+ * it under the terms of the GNU Lesser General Public License as published by
+ * the Free Software Foundation, either version 3 of the License, or
+ * (at your option) any later version.
+ *
+ * FlutterFFmpeg is distributed in the hope that it will be useful,
+ * but WITHOUT ANY WARRANTY; without even the implied warranty of
+ * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
+ * GNU Lesser General Public License for more details.
+ *
+ * You should have received a copy of the GNU Lesser General Public License
+ * along with FlutterFFmpeg.  If not, see <http://www.gnu.org/licenses/>.
+ */
+
+#import "FlutterFFmpegPlugin.h"
+
+#import <mobileffmpeg/ArchDetect.h>
+#import <mobileffmpeg/MobileFFmpeg.h>
+#import <mobileffmpeg/MobileFFmpegConfig.h>
+
+static NSString *const PLATFORM_NAME = @"ios";
+
+static NSString *const KEY_VERSION = @"version";
+static NSString *const KEY_RC = @"rc";
+static NSString *const KEY_PLATFORM = @"platform";
+static NSString *const KEY_PACKAGE_NAME = @"packageName";
+static NSString *const KEY_LAST_RC = @"lastRc";
+static NSString *const KEY_LAST_COMMAND_OUTPUT = @"lastCommandOutput";
+
+static NSString *const KEY_LOG_TEXT = @"log";
+static NSString *const KEY_LOG_LEVEL = @"level";
+
+static NSString *const KEY_STAT_TIME = @"time";
+static NSString *const KEY_STAT_SIZE = @"size";
+static NSString *const KEY_STAT_BITRATE = @"bitrate";
+static NSString *const KEY_STAT_SPEED = @"speed";
+static NSString *const KEY_STAT_VIDEO_FRAME_NUMBER = @"videoFrameNumber";
+static NSString *const KEY_STAT_VIDEO_QUALITY = @"videoQuality";
+static NSString *const KEY_STAT_VIDEO_FPS = @"videoFps";
+
+static NSString *const EVENT_LOG = @"FlutterFFmpegLogCallback";
+static NSString *const EVENT_STAT = @"FlutterFFmpegStatisticsCallback";
+
+/**
+ * Flutter FFmpeg Plugin
+ */
+@implementation FlutterFFmpegPlugin {
+    FlutterEventSink _eventSink;
+}
+
+- (FlutterError *)onListenWithArguments:(id)arguments eventSink:(FlutterEventSink)eventSink {
+    _eventSink = eventSink;
+    return nil;
+}
+
+- (FlutterError *)onCancelWithArguments:(id)arguments {
+    _eventSink = nil;
+    return nil;
+}
+
++ (void)registerWithRegistrar:(NSObject<FlutterPluginRegistrar>*)registrar {
+    FlutterFFmpegPlugin* instance = [[FlutterFFmpegPlugin alloc] init];
+
+    FlutterMethodChannel* methodChannel = [FlutterMethodChannel methodChannelWithName:@"flutter_ffmpeg" binaryMessenger:[registrar messenger]];
+    [registrar addMethodCallDelegate:instance channel:methodChannel];
+
+    FlutterEventChannel* eventChannel = [FlutterEventChannel eventChannelWithName:@"flutter_ffmpeg_event" binaryMessenger:[registrar messenger]];
+    [eventChannel setStreamHandler:instance];
+}
+
+- (void)handleMethodCall:(FlutterMethodCall*)call result:(FlutterResult)result {
+
+    // ARGUMENTS
+    NSArray* arguments = call.arguments[@"arguments"];
+    NSString* command = call.arguments[@"command"];
+    NSString* delimiter = call.arguments[@"delimiter"];
+
+    if ([@"getPlatform" isEqualToString:call.method]) {
+
+        NSString *architecture = [ArchDetect getArch];
+        result([FlutterFFmpegPlugin toStringDictionary:KEY_PLATFORM :[NSString stringWithFormat:@"%@-%@", PLATFORM_NAME, architecture]]);
+
+    } else if ([@"getFFmpegVersion" isEqualToString:call.method]) {
+
+        result([FlutterFFmpegPlugin toStringDictionary:KEY_VERSION :[MobileFFmpeg getFFmpegVersion]]);
+
+    } else if ([@"executeWithArguments" isEqualToString:call.method]) {
+
+        NSLog(@"Running FFmpeg with arguments: %@.\n", arguments);
+
+        dispatch_async(dispatch_get_global_queue(DISPATCH_QUEUE_PRIORITY_DEFAULT, 0), ^{
+
+            int rc = [MobileFFmpeg executeWithArguments:arguments];
+
+            NSLog(@"FFmpeg exited with rc: %d\n", rc);
+
+            result([FlutterFFmpegPlugin toIntDictionary:KEY_RC :[NSNumber numberWithInt:rc]]);
+        });
+
+    } else if ([@"execute" isEqualToString:call.method]) {
+
+        if (delimiter == nil) {
+            delimiter = @" ";
+        }
+
+        NSLog(@"Running FFmpeg command: %@ with delimiter %@.\n", command, delimiter);
+
+        dispatch_async(dispatch_get_global_queue(DISPATCH_QUEUE_PRIORITY_DEFAULT, 0), ^{
+            int rc = [MobileFFmpeg execute:command delimiter:delimiter];
+
+            NSLog(@"FFmpeg exited with rc: %d\n", rc);
+
+            result([FlutterFFmpegPlugin toIntDictionary:KEY_RC :[NSNumber numberWithInt:rc]]);
+        });
+
+    } else if ([@"cancel" isEqualToString:call.method]) {
+
+        [MobileFFmpeg cancel];
+
+    } else if ([@"enableRedirection" isEqualToString:call.method]) {
+
+        [MobileFFmpegConfig enableRedirection];
+
+    } else if ([@"disableRedirection" isEqualToString:call.method]) {
+
+        [MobileFFmpegConfig disableRedirection];
+
+    } else if ([@"getLogLevel" isEqualToString:call.method]) {
+
+        int logLevel = [MobileFFmpegConfig getLogLevel];
+        result([FlutterFFmpegPlugin toIntDictionary:KEY_LOG_LEVEL :[NSNumber numberWithInt:logLevel]]);
+
+    } else if ([@"setLogLevel" isEqualToString:call.method]) {
+
+        NSNumber* logLevel = call.arguments[@"level"];
+        [MobileFFmpegConfig setLogLevel:[logLevel intValue]];
+
+    } else if ([@"enableLogs" isEqualToString:call.method]) {
+
+        [MobileFFmpegConfig setLogDelegate:self];
+
+    } else if ([@"disableLogs" isEqualToString:call.method]) {
+
+        [MobileFFmpegConfig setLogDelegate:nil];
+
+    } else if ([@"enableStatistics" isEqualToString:call.method]) {
+
+        [MobileFFmpegConfig setStatisticsDelegate:self];
+
+    } else if ([@"disableStatistics" isEqualToString:call.method]) {
+
+        [MobileFFmpegConfig setStatisticsDelegate:nil];
+
+    } else if ([@"getLastReceivedStatistics" isEqualToString:call.method]) {
+
+        Statistics *statistics = [MobileFFmpegConfig getLastReceivedStatistics];
+        result([FlutterFFmpegPlugin toStatisticsDictionary:statistics]);
+
+    } else if ([@"resetStatistics" isEqualToString:call.method]) {
+
+        [MobileFFmpegConfig resetStatistics];
+
+    } else if ([@"setFontconfigConfigurationPath" isEqualToString:call.method]) {
+
+        NSString* path = call.arguments[@"path"];
+        [MobileFFmpegConfig setFontconfigConfigurationPath:path];
+
+    } else if ([@"setFontDirectory" isEqualToString:call.method]) {
+
+        NSString* fontDirectoryPath = call.arguments[@"fontDirectory"];
+        NSDictionary* fontNameMapping = call.arguments[@"fontNameMap"];
+        [MobileFFmpegConfig setFontDirectory:fontDirectoryPath with:fontNameMapping];
+
+    } else if ([@"getPackageName" isEqualToString:call.method]) {
+
+        NSString *packageName = [MobileFFmpegConfig getPackageName];
+        result([FlutterFFmpegPlugin toStringDictionary:KEY_PACKAGE_NAME :packageName]);
+
+    } else if ([@"getExternalLibraries" isEqualToString:call.method]) {
+
+        NSArray *externalLibraries = [MobileFFmpegConfig getExternalLibraries];
+        result(externalLibraries);
+
+    } else if ([@"getLastReturnCode" isEqualToString:call.method]) {
+
+        int lastReturnCode = [MobileFFmpeg getLastReturnCode];
+        result([FlutterFFmpegPlugin toIntDictionary:KEY_LAST_RC :[NSNumber numberWithInt:lastReturnCode]]);
+
+    } else if ([@"getLastCommandOutput" isEqualToString:call.method]) {
+
+        NSString *lastCommandOutput = [MobileFFmpeg getLastCommandOutput];
+        result([FlutterFFmpegPlugin toStringDictionary:KEY_LAST_COMMAND_OUTPUT :lastCommandOutput]);
+
+    } else if ([@"getMediaInformation" isEqualToString:call.method]) {
+
+        NSString* path = call.arguments[@"path"];
+        NSNumber* timeout = call.arguments[@"timeout"];
+
+        NSLog(@"Getting media information for %@ with timeout %d.\n", path, [timeout intValue]);
+
+        dispatch_async(dispatch_get_global_queue(DISPATCH_QUEUE_PRIORITY_DEFAULT, 0), ^{
+            MediaInformation *mediaInformation = [MobileFFmpeg getMediaInformation:path timeout:[timeout intValue]];
+            result([FlutterFFmpegPlugin toMediaInformationDictionary:mediaInformation]);
+        });
+
+    } else {
+
+        result(FlutterMethodNotImplemented);
+
+    }
+}
+
+- (void)logCallback: (int)level :(NSString*)message {
+    dispatch_async(dispatch_get_main_queue(), ^{
+        NSMutableDictionary *dictionary = [[NSMutableDictionary alloc] init];
+        dictionary[KEY_LOG_LEVEL] = [NSNumber numberWithInt:level];
+        dictionary[KEY_LOG_TEXT] = message;
+
+        [self emitLogMessage: dictionary];
+    });
+}
+
+- (void)statisticsCallback:(Statistics *)statistics {
+    dispatch_async(dispatch_get_main_queue(), ^{
+        [self emitStatistics: statistics];
+    });
+}
+
+- (void)emitLogMessage:(NSDictionary*)logMessage{
+    _eventSink([FlutterFFmpegPlugin toStringDictionary:EVENT_LOG :logMessage]);
+}
+
+- (void)emitStatistics:(Statistics*)statistics{
+    NSDictionary *dictionary = [FlutterFFmpegPlugin toStatisticsDictionary:statistics];
+    _eventSink([FlutterFFmpegPlugin toStringDictionary:EVENT_STAT :dictionary]);
+}
+
++ (NSDictionary *)toStringDictionary:(NSString*)key :(NSString*)value {
+    NSMutableDictionary *dictionary = [[NSMutableDictionary alloc] init];
+    dictionary[key] = value;
+
+    return dictionary;
+}
+
++ (NSDictionary *)toIntDictionary:(NSString*)key :(NSNumber*)value {
+    NSMutableDictionary *dictionary = [[NSMutableDictionary alloc] init];
+    dictionary[key] = value;
+
+    return dictionary;
+}
+
++ (NSDictionary *)toStatisticsDictionary:(Statistics*)statistics {
+    NSMutableDictionary *dictionary = [[NSMutableDictionary alloc] init];
+
+    if (statistics != nil) {
+        dictionary[KEY_STAT_TIME] = [NSNumber numberWithInt: [statistics getTime]];
+        dictionary[KEY_STAT_SIZE] = [NSNumber numberWithLong: [statistics getSize]];
+
+        dictionary[KEY_STAT_BITRATE] = [NSNumber numberWithDouble: [statistics getBitrate]];
+        dictionary[KEY_STAT_SPEED] = [NSNumber numberWithDouble: [statistics getSpeed]];
+
+        dictionary[KEY_STAT_VIDEO_FRAME_NUMBER] = [NSNumber numberWithInt: [statistics getVideoFrameNumber]];
+        dictionary[KEY_STAT_VIDEO_QUALITY] = [NSNumber numberWithFloat: [statistics getVideoQuality]];
+        dictionary[KEY_STAT_VIDEO_FPS] = [NSNumber numberWithFloat: [statistics getVideoFps]];
+    }
+
+    return dictionary;
+}
+
++ (NSDictionary *)toMediaInformationDictionary:(MediaInformation*)mediaInformation {
+    NSMutableDictionary *dictionary = [[NSMutableDictionary alloc] init];
+
+    if (mediaInformation != nil) {
+        if ([mediaInformation getFormat] != nil) {
+            dictionary[@"format"] =  [mediaInformation getFormat];
+        }
+        if ([mediaInformation getPath] != nil) {
+            dictionary[@"path"] = [mediaInformation getPath];
+        }
+        if ([mediaInformation getStartTime] != nil) {
+            dictionary[@"startTime"] = [mediaInformation getStartTime];
+        }
+        if ([mediaInformation getDuration] != nil) {
+            dictionary[@"duration"] = [mediaInformation getDuration];
+        }
+        if ([mediaInformation getBitrate] != nil) {
+            dictionary[@"bitrate"] = [mediaInformation getBitrate];
+        }
+        if ([mediaInformation getRawInformation] != nil) {
+            dictionary[@"rawInformation"] = [mediaInformation getRawInformation];
+        }
+
+        NSDictionary *metadataDictionary = [mediaInformation getMetadataEntries];
+        if (metadataDictionary != nil && ([metadataDictionary count] > 0)) {
+            dictionary[@"metadata"] = metadataDictionary;
+        }
+
+        NSArray *streams = [mediaInformation getStreams];
+        if (streams != nil && ([streams count] > 0)) {
+            NSMutableArray *array = [[NSMutableArray alloc] init];
+
+            for (int i=0; i < [streams count]; i++) {
+                StreamInformation *streamInformation= [streams objectAtIndex:i];
+                [array addObject: [FlutterFFmpegPlugin toStreamInformationDictionary:streamInformation]];
+            }
+
+            dictionary[@"streams"] = array;
+        }
+    }
+
+    return dictionary;
+}
+
++ (NSDictionary *)toStreamInformationDictionary:(StreamInformation*)streamInformation {
+    NSMutableDictionary *dictionary = [[NSMutableDictionary alloc] init];
+
+    if (streamInformation != nil) {
+        if ([streamInformation getIndex] != nil) {
+            dictionary[@"index"] = [streamInformation getIndex];
+        }
+        if ([streamInformation getType] != nil) {
+            dictionary[@"type"] = [streamInformation getType];
+        }
+        if ([streamInformation getCodec] != nil) {
+            dictionary[@"codec"] = [streamInformation getCodec];
+        }
+        if ([streamInformation getFullCodec] != nil) {
+            dictionary[@"fullCodec"] = [streamInformation getFullCodec];
+        }
+        if ([streamInformation getFormat] != nil) {
+            dictionary[@"format"] = [streamInformation getFormat];
+        }
+        if ([streamInformation getFullFormat] != nil) {
+            dictionary[@"fullFormat"] = [streamInformation getFullFormat];
+        }
+        if ([streamInformation getWidth] != nil) {
+            dictionary[@"width"] = [streamInformation getWidth];
+        }
+        if ([streamInformation getHeight] != nil) {
+            dictionary[@"height"] = [streamInformation getHeight];
+        }
+        if ([streamInformation getBitrate] != nil) {
+            dictionary[@"bitrate"] = [streamInformation getBitrate];
+        }
+        if ([streamInformation getSampleRate] != nil) {
+            dictionary[@"sampleRate"] = [streamInformation getSampleRate];
+        }
+        if ([streamInformation getSampleFormat] != nil) {
+            dictionary[@"sampleFormat"] = [streamInformation getSampleFormat];
+        }
+        if ([streamInformation getChannelLayout] != nil) {
+            dictionary[@"channelLayout"] = [streamInformation getChannelLayout];
+        }
+        if ([streamInformation getSampleAspectRatio] != nil) {
+            dictionary[@"sampleAspectRatio"] = [streamInformation getSampleAspectRatio];
+        }
+        if ([streamInformation getDisplayAspectRatio] != nil) {
+            dictionary[@"displayAspectRatio"] = [streamInformation getDisplayAspectRatio];
+        }
+        if ([streamInformation getAverageFrameRate] != nil) {
+            dictionary[@"averageFrameRate"] = [streamInformation getAverageFrameRate];
+        }
+        if ([streamInformation getRealFrameRate] != nil) {
+            dictionary[@"realFrameRate"] = [streamInformation getRealFrameRate];
+        }
+        if ([streamInformation getTimeBase] != nil) {
+            dictionary[@"timeBase"] = [streamInformation getTimeBase];
+        }
+        if ([streamInformation getCodecTimeBase] != nil) {
+            dictionary[@"codecTimeBase"] = [streamInformation getCodecTimeBase];
+        }
+
+        NSDictionary *metadataDictionary = [streamInformation getMetadataEntries];
+        if (metadataDictionary != nil && ([metadataDictionary count] > 0)) {
+            dictionary[@"metadata"] = metadataDictionary;
+        }
+    }
+
+    return dictionary;
+}
+
+@end

+ 22 - 0
packages/flutter_ffmpeg_audio/ios/flutter_ffmpeg.podspec

@@ -0,0 +1,22 @@
+Pod::Spec.new do |s|
+  s.name             = 'flutter_ffmpeg'
+  s.version          = '0.1.0'
+  s.summary          = 'FFmpeg plugin for Flutter.'
+  s.description      = 'FFmpeg plugin based on mobile-ffmpeg for Flutter.'
+  s.homepage         = 'https://github.com/tanersener/flutter-ffmpeg'
+
+  s.author           = { 'Taner Sener' => 'tanersener@gmail.com' }
+  s.license          = { :file => '../LICENSE' }
+
+  s.requires_arc     = true
+  s.ios.deployment_target = '9.3'
+
+  s.source              = { :path => '.' }
+  s.source_files        = 'Classes/**/*'
+  s.public_header_files = 'Classes/**/*.h'
+
+  s.dependency      'Flutter'
+  s.dependency      'mobile-ffmpeg-audio', '4.2.LTS'
+
+end
+

+ 352 - 0
packages/flutter_ffmpeg_audio/lib/flutter_ffmpeg.dart

@@ -0,0 +1,352 @@
+/*
+ * Copyright (c) 2019 Taner Sener
+ *
+ * This file is part of FlutterFFmpeg.
+ *
+ * FlutterFFmpeg is free software: you can redistribute it and/or modify
+ * it under the terms of the GNU Lesser General Public License as published by
+ * the Free Software Foundation, either version 3 of the License, or
+ * (at your option) any later version.
+ *
+ * FlutterFFmpeg is distributed in the hope that it will be useful,
+ * but WITHOUT ANY WARRANTY; without even the implied warranty of
+ * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
+ * GNU Lesser General Public License for more details.
+ *
+ * You should have received a copy of the GNU Lesser General Public License
+ * along with FlutterFFmpeg.  If not, see <http://www.gnu.org/licenses/>.
+ */
+
+import 'dart:async';
+
+import 'package:flutter/services.dart';
+
+class FlutterFFmpeg {
+  static const MethodChannel _methodChannel = const MethodChannel('flutter_ffmpeg');
+  static const EventChannel _eventChannel = const EventChannel('flutter_ffmpeg_event');
+
+  Function(int level, String message) logCallback;
+  Function(int time, int size, double bitrate, double speed, int videoFrameNumber, double videoQuality, double videoFps) statisticsCallback;
+
+  FlutterFFmpeg() {
+    logCallback = null;
+    statisticsCallback = null;
+
+    print("Loading flutter-ffmpeg.");
+
+    _eventChannel.receiveBroadcastStream().listen(_onEvent, onError: _onError);
+
+    enableLogs();
+    enableStatistics();
+    enableRedirection();
+
+    getPlatform().then((name) => print("Loaded flutter-ffmpeg-$name."));
+  }
+
+  void _onEvent(Object event) {
+    if (event is Map<dynamic, dynamic>) {
+      final Map<String, dynamic> eventMap = event.cast();
+      final Map<dynamic, dynamic> logEvent = eventMap['FlutterFFmpegLogCallback'];
+      final Map<dynamic, dynamic> statisticsEvent = eventMap['FlutterFFmpegStatisticsCallback'];
+
+      if (logEvent != null) {
+        int level = logEvent['level'];
+        String message = logEvent['log'];
+
+        if (this.logCallback == null) {
+          if (message.length > 0) {
+            // PRINT ALREADY ADDS NEW LINE. SO REMOVE THIS ONE
+            if (message.endsWith('\n')) {
+              print(message.substring(0, message.length - 1));
+            } else {
+              print(message);
+            }
+          }
+        } else {
+          this.logCallback(level, message);
+        }
+      }
+
+      if (statisticsEvent != null) {
+        if (this.statisticsCallback != null) {
+          int time = statisticsEvent['time'];
+          int size = statisticsEvent['size'];
+          double bitrate = _doublePrecision(statisticsEvent['bitrate'], 2);
+          double speed = _doublePrecision(statisticsEvent['speed'], 2);
+          int videoFrameNumber = statisticsEvent['videoFrameNumber'];
+          double videoQuality = _doublePrecision(statisticsEvent['videoQuality'], 2);
+          double videoFps = _doublePrecision(statisticsEvent['videoFps'], 2);
+
+          this.statisticsCallback(time, size, bitrate, speed, videoFrameNumber, videoQuality, videoFps);
+        }
+      }
+    }
+  }
+
+  void _onError(Object error) {
+    print('Event error: $error');
+  }
+
+  double _doublePrecision(double value, int precision) {
+    if (value == null) {
+      return 0;
+    } else {
+      return num.parse(value.toStringAsFixed(precision));
+    }
+  }
+
+  /// Returns FFmpeg version bundled within the library.
+  Future<String> getFFmpegVersion() async {
+    try {
+      final Map<dynamic, dynamic> result = await _methodChannel.invokeMethod('getFFmpegVersion');
+      return result['version'];
+    } on PlatformException catch (e) {
+      print("Plugin error: ${e.message}");
+      return null;
+    }
+  }
+
+  /// Returns platform name where library is loaded.
+  Future<String> getPlatform() async {
+    try {
+      final Map<dynamic, dynamic> result = await _methodChannel.invokeMethod('getPlatform');
+      return result['platform'];
+    } on PlatformException catch (e) {
+      print("Plugin error: ${e.message}");
+      return null;
+    }
+  }
+
+  /// Executes FFmpeg with [commandArguments] provided.
+  Future<int> executeWithArguments(List<String> arguments) async {
+    try {
+      final Map<dynamic, dynamic> result = await _methodChannel.invokeMethod('executeWithArguments', {'arguments': arguments});
+      return result['rc'];
+    } on PlatformException catch (e) {
+      print("Plugin error: ${e.message}");
+      return -1;
+    }
+  }
+
+  /// Executes FFmpeg [command] provided. Command is split into arguments using provided [delimiter].
+  Future<int> execute(String command, [String delimiter = ' ']) async {
+    try {
+      final Map<dynamic, dynamic> result = await _methodChannel.invokeMethod('execute', {'command': command, 'delimiter': delimiter});
+      return result['rc'];
+    } on PlatformException catch (e) {
+      print("Plugin error: ${e.message}");
+      return -1;
+    }
+  }
+
+  /// Cancels an ongoing operation.
+  Future<void> cancel() async {
+    try {
+      await _methodChannel.invokeMethod('cancel');
+    } on PlatformException catch (e) {
+      print("Plugin error: ${e.message}");
+    }
+  }
+
+  /// Enables redirection
+  Future<void> enableRedirection() async {
+    try {
+      await _methodChannel.invokeMethod('enableRedirection');
+    } on PlatformException catch (e) {
+      print("Plugin error: ${e.message}");
+    }
+  }
+
+  /// Disables log and statistics redirection. By default redirection is enabled in constructor.
+  /// When redirection is enabled FFmpeg logs are printed to console and can be routed further to a callback function.
+  /// By disabling redirection, logs are redirected to stderr.
+  /// Statistics redirection behaviour is similar. Statistics are not printed at all if redirection is not enabled.
+  /// If it is enabled then it is possible to define a statistics callback function but if you don't, they are not
+  /// printed anywhere and only saved as codelastReceivedStatistics data which can be polled with
+  /// [getLastReceivedStatistics()].
+  Future<void> disableRedirection() async {
+    try {
+      await _methodChannel.invokeMethod('disableRedirection');
+    } on PlatformException catch (e) {
+      print("Plugin error: ${e.message}");
+    }
+  }
+
+  /// Returns log level.
+  Future<int> getLogLevel() async {
+    try {
+      final Map<dynamic, dynamic> result = await _methodChannel.invokeMethod('getLogLevel');
+      return result['level'];
+    } on PlatformException catch (e) {
+      print("Plugin error: ${e.message}");
+      return -1;
+    }
+  }
+
+  /// Sets log level.
+  Future<void> setLogLevel(int logLevel) async {
+    try {
+      await _methodChannel.invokeMethod('setLogLevel', {'level': logLevel});
+    } on PlatformException catch (e) {
+      print("Plugin error: ${e.message}");
+    }
+  }
+
+  /// Enables log events
+  Future<void> enableLogs() async {
+    try {
+      await _methodChannel.invokeMethod('enableLogs');
+    } on PlatformException catch (e) {
+      print("Plugin error: ${e.message}");
+    }
+  }
+
+  /// Disables log functionality of the library. Logs will not be printed to console and log callback will be disabled.
+  /// Note that log functionality is enabled by default.
+  Future<void> disableLogs() async {
+    try {
+      await _methodChannel.invokeMethod('disableLogs');
+    } on PlatformException catch (e) {
+      print("Plugin error: ${e.message}");
+    }
+  }
+
+  /// Enables statistics events.
+  Future<void> enableStatistics() async {
+    try {
+      await _methodChannel.invokeMethod('enableStatistics');
+    } on PlatformException catch (e) {
+      print("Plugin error: ${e.message}");
+    }
+  }
+
+  /// Disables statistics functionality of the library. Statistics callback will be disabled but the last received
+  /// statistics data will be still available.
+  /// Note that statistics functionality is enabled by default.
+  Future<void> disableStatistics() async {
+    try {
+      await _methodChannel.invokeMethod('disableStatistics');
+    } on PlatformException catch (e) {
+      print("Plugin error: ${e.message}");
+    }
+  }
+
+  /// Sets a callback to redirect FFmpeg logs. [newCallback] is a new log callback function, use null to disable a previously defined callback
+  void enableLogCallback(Function(int level, String message) newCallback) {
+    try {
+      this.logCallback = newCallback;
+    } on PlatformException catch (e) {
+      print("Plugin error: ${e.message}");
+    }
+  }
+
+  /// Sets a callback to redirect FFmpeg statistics. [newCallback] is a new statistics callback function, use null to disable a previously defined callback
+  void enableStatisticsCallback(Function(int time, int size, double bitrate, double speed, int videoFrameNumber, double videoQuality, double videoFps) newCallback) {
+    try {
+      this.statisticsCallback = newCallback;
+    } on PlatformException catch (e) {
+      print("Plugin error: ${e.message}");
+    }
+  }
+
+  /// Returns the last received statistics data stored in bitrate, size, speed, time, videoFps, videoFrameNumber and
+  /// videoQuality fields
+  Future<Map<dynamic, dynamic>> getLastReceivedStatistics() async {
+    try {
+      final Map<dynamic, dynamic> result = await _methodChannel.invokeMethod('getLastReceivedStatistics');
+      return result;
+    } on PlatformException catch (e) {
+      print("Plugin error: ${e.message}");
+      return null;
+    }
+  }
+
+  /// Resets last received statistics. It is recommended to call it before starting a new execution.
+  Future<void> resetStatistics() async {
+    try {
+      await _methodChannel.invokeMethod('resetStatistics');
+    } on PlatformException catch (e) {
+      print("Plugin error: ${e.message}");
+    }
+  }
+
+  /// Sets and overrides fontconfig configuration directory.
+  Future<void> setFontconfigConfigurationPath(String path) async {
+    try {
+      await _methodChannel.invokeMethod('setFontconfigConfigurationPath', {'path': path});
+    } on PlatformException catch (e) {
+      print("Plugin error: ${e.message}");
+    }
+  }
+
+  /// Registers fonts inside the given [fontDirectory], so they are available to use in FFmpeg filters.
+  Future<void> setFontDirectory(String fontDirectory, Map<String, String> fontNameMap) async {
+    var parameters;
+    if (fontNameMap == null) {
+      parameters = {'fontDirectory': fontDirectory};
+    } else {
+      parameters = {'fontDirectory': fontDirectory, 'fontNameMap': fontNameMap};
+    }
+
+    try {
+      await _methodChannel.invokeMethod('setFontDirectory', parameters);
+    } on PlatformException catch (e) {
+      print("Plugin error: ${e.message}");
+    }
+  }
+
+  /// Returns FlutterFFmpeg package name.
+  Future<String> getPackageName() async {
+    try {
+      final Map<dynamic, dynamic> result = await _methodChannel.invokeMethod('getPackageName');
+      return result['packageName'];
+    } on PlatformException catch (e) {
+      print("Plugin error: ${e.message}");
+      return null;
+    }
+  }
+
+  /// Returns supported external libraries.
+  Future<List<dynamic>> getExternalLibraries() async {
+    try {
+      final List<dynamic> result = await _methodChannel.invokeMethod('getExternalLibraries');
+      return result;
+    } on PlatformException catch (e) {
+      print("Plugin error: ${e.message}");
+      return null;
+    }
+  }
+
+  /// Returns return code of last executed command.
+  Future<int> getLastReturnCode() async {
+    try {
+      final Map<dynamic, dynamic> result = await _methodChannel.invokeMethod('getLastReturnCode');
+      return result['lastRc'];
+    } on PlatformException catch (e) {
+      print("Plugin error: ${e.message}");
+      return -1;
+    }
+  }
+
+  /// Returns log output of last executed command. Please note that disabling redirection using
+  /// [disableRedirection()] method also disables this functionality.
+  Future<String> getLastCommandOutput() async {
+    try {
+      final Map<dynamic, dynamic> result = await _methodChannel.invokeMethod('getLastCommandOutput');
+      return result['lastCommandOutput'];
+    } on PlatformException catch (e) {
+      print("Plugin error: ${e.message}");
+      return null;
+    }
+  }
+
+  /// Returns media information for given [path] using optional [timeout]
+  Future<Map<dynamic, dynamic>> getMediaInformation(String path, [int timeout = 10000]) async {
+    try {
+      return await _methodChannel.invokeMethod('getMediaInformation', {'path': path, 'timeout': timeout});
+    } on PlatformException catch (e) {
+      print("Plugin error: ${e.message}");
+      return null;
+    }
+  }
+}

+ 76 - 0
packages/flutter_ffmpeg_audio/lib/log_level.dart

@@ -0,0 +1,76 @@
+/*
+ * Copyright (c) 2019 Taner Sener
+ *
+ * This file is part of FlutterFFmpeg.
+ *
+ * FlutterFFmpeg is free software: you can redistribute it and/or modify
+ * it under the terms of the GNU Lesser General Public License as published by
+ * the Free Software Foundation, either version 3 of the License, or
+ * (at your option) any later version.
+ *
+ * FlutterFFmpeg is distributed in the hope that it will be useful,
+ * but WITHOUT ANY WARRANTY; without even the implied warranty of
+ * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
+ * GNU Lesser General Public License for more details.
+ *
+ * You should have received a copy of the GNU Lesser General Public License
+ * along with FlutterFFmpeg.  If not, see <http://www.gnu.org/licenses/>.
+ */
+
+class LogLevel {
+  /// Print no output.
+  static const int AV_LOG_QUIET = -8;
+
+  /// Something went really wrong and we will crash now.
+  static const int AV_LOG_PANIC = 0;
+
+  /// Something went wrong and recovery is not possible.
+  /// For example, no header was found for a format which depends
+  /// on headers or an illegal combination of parameters is used.
+  static const int AV_LOG_FATAL = 8;
+
+  /// Something went wrong and cannot losslessly be recovered.
+  /// However, not all future data is affected.
+  static const int AV_LOG_ERROR = 16;
+
+  /// Something somehow does not look correct. This may or may not
+  /// lead to problems. An example would be the use of '-vstrict -2'.
+  static const int AV_LOG_WARNING = 24;
+
+  /// int Standard information.
+  static const int AV_LOG_INFO = 32;
+
+  /// Detailed information.
+  static const int AV_LOG_VERBOSE = 40;
+
+  /// Stuff which is only useful for libav* developers.
+  static const int AV_LOG_DEBUG = 48;
+
+  /// Extremely verbose debugging, useful for libav* development.
+  static const int AV_LOG_TRACE = 56;
+
+  /// Returns log level string from int
+  static String levelToString(int level) {
+    switch (level) {
+      case LogLevel.AV_LOG_TRACE:
+        return "TRACE";
+      case LogLevel.AV_LOG_DEBUG:
+        return "DEBUG";
+      case LogLevel.AV_LOG_VERBOSE:
+        return "VERBOSE";
+      case LogLevel.AV_LOG_INFO:
+        return "INFO";
+      case LogLevel.AV_LOG_WARNING:
+        return "WARNING";
+      case LogLevel.AV_LOG_ERROR:
+        return "ERROR";
+      case LogLevel.AV_LOG_FATAL:
+        return "FATAL";
+      case LogLevel.AV_LOG_PANIC:
+        return "PANIC";
+      case LogLevel.AV_LOG_QUIET:
+      default:
+        return "";
+    }
+  }
+}

+ 17 - 0
packages/flutter_ffmpeg_audio/pubspec.yaml

@@ -0,0 +1,17 @@
+name: flutter_ffmpeg
+description: FFmpeg plugin for Flutter.
+version: 0.1.0
+author: Taner Sener <tanersener@gmail.com>
+homepage: https://github.com/tanersener/flutter-ffmpeg
+
+environment:
+  sdk: ">=2.0.0-dev.68.0 <3.0.0"
+
+dependencies:
+  flutter:
+    sdk: flutter
+
+flutter:
+  plugin:
+    androidPackage: com.arthenica.flutter.ffmpeg
+    pluginClass: FlutterFFmpegPlugin

+ 7 - 0
packages/flutter_ffmpeg_full-gpl/.gitignore

@@ -0,0 +1,7 @@
+.idea/
+.DS_Store
+.dart_tool/
+.pub/
+build/
+pubspec.lock
+.packages

+ 10 - 0
packages/flutter_ffmpeg_full-gpl/.metadata

@@ -0,0 +1,10 @@
+# This file tracks properties of this Flutter project.
+# Used by Flutter tool to assess capabilities and perform upgrades etc.
+#
+# This file should be version controlled and should not be manually edited.
+
+version:
+  revision: 5391447fae6209bb21a89e6a5a6583cac1af9b4b
+  channel: stable
+
+project_type: plugin

+ 674 - 0
packages/flutter_ffmpeg_full-gpl/LICENSE

@@ -0,0 +1,674 @@
+                    GNU GENERAL PUBLIC LICENSE
+                       Version 3, 29 June 2007
+
+ Copyright (C) 2007 Free Software Foundation, Inc. <http://fsf.org/>
+ Everyone is permitted to copy and distribute verbatim copies
+ of this license document, but changing it is not allowed.
+
+                            Preamble
+
+  The GNU General Public License is a free, copyleft license for
+software and other kinds of works.
+
+  The licenses for most software and other practical works are designed
+to take away your freedom to share and change the works.  By contrast,
+the GNU General Public License is intended to guarantee your freedom to
+share and change all versions of a program--to make sure it remains free
+software for all its users.  We, the Free Software Foundation, use the
+GNU General Public License for most of our software; it applies also to
+any other work released this way by its authors.  You can apply it to
+your programs, too.
+
+  When we speak of free software, we are referring to freedom, not
+price.  Our General Public Licenses are designed to make sure that you
+have the freedom to distribute copies of free software (and charge for
+them if you wish), that you receive source code or can get it if you
+want it, that you can change the software or use pieces of it in new
+free programs, and that you know you can do these things.
+
+  To protect your rights, we need to prevent others from denying you
+these rights or asking you to surrender the rights.  Therefore, you have
+certain responsibilities if you distribute copies of the software, or if
+you modify it: responsibilities to respect the freedom of others.
+
+  For example, if you distribute copies of such a program, whether
+gratis or for a fee, you must pass on to the recipients the same
+freedoms that you received.  You must make sure that they, too, receive
+or can get the source code.  And you must show them these terms so they
+know their rights.
+
+  Developers that use the GNU GPL protect your rights with two steps:
+(1) assert copyright on the software, and (2) offer you this License
+giving you legal permission to copy, distribute and/or modify it.
+
+  For the developers' and authors' protection, the GPL clearly explains
+that there is no warranty for this free software.  For both users' and
+authors' sake, the GPL requires that modified versions be marked as
+changed, so that their problems will not be attributed erroneously to
+authors of previous versions.
+
+  Some devices are designed to deny users access to install or run
+modified versions of the software inside them, although the manufacturer
+can do so.  This is fundamentally incompatible with the aim of
+protecting users' freedom to change the software.  The systematic
+pattern of such abuse occurs in the area of products for individuals to
+use, which is precisely where it is most unacceptable.  Therefore, we
+have designed this version of the GPL to prohibit the practice for those
+products.  If such problems arise substantially in other domains, we
+stand ready to extend this provision to those domains in future versions
+of the GPL, as needed to protect the freedom of users.
+
+  Finally, every program is threatened constantly by software patents.
+States should not allow patents to restrict development and use of
+software on general-purpose computers, but in those that do, we wish to
+avoid the special danger that patents applied to a free program could
+make it effectively proprietary.  To prevent this, the GPL assures that
+patents cannot be used to render the program non-free.
+
+  The precise terms and conditions for copying, distribution and
+modification follow.
+
+                       TERMS AND CONDITIONS
+
+  0. Definitions.
+
+  "This License" refers to version 3 of the GNU General Public License.
+
+  "Copyright" also means copyright-like laws that apply to other kinds of
+works, such as semiconductor masks.
+
+  "The Program" refers to any copyrightable work licensed under this
+License.  Each licensee is addressed as "you".  "Licensees" and
+"recipients" may be individuals or organizations.
+
+  To "modify" a work means to copy from or adapt all or part of the work
+in a fashion requiring copyright permission, other than the making of an
+exact copy.  The resulting work is called a "modified version" of the
+earlier work or a work "based on" the earlier work.
+
+  A "covered work" means either the unmodified Program or a work based
+on the Program.
+
+  To "propagate" a work means to do anything with it that, without
+permission, would make you directly or secondarily liable for
+infringement under applicable copyright law, except executing it on a
+computer or modifying a private copy.  Propagation includes copying,
+distribution (with or without modification), making available to the
+public, and in some countries other activities as well.
+
+  To "convey" a work means any kind of propagation that enables other
+parties to make or receive copies.  Mere interaction with a user through
+a computer network, with no transfer of a copy, is not conveying.
+
+  An interactive user interface displays "Appropriate Legal Notices"
+to the extent that it includes a convenient and prominently visible
+feature that (1) displays an appropriate copyright notice, and (2)
+tells the user that there is no warranty for the work (except to the
+extent that warranties are provided), that licensees may convey the
+work under this License, and how to view a copy of this License.  If
+the interface presents a list of user commands or options, such as a
+menu, a prominent item in the list meets this criterion.
+
+  1. Source Code.
+
+  The "source code" for a work means the preferred form of the work
+for making modifications to it.  "Object code" means any non-source
+form of a work.
+
+  A "Standard Interface" means an interface that either is an official
+standard defined by a recognized standards body, or, in the case of
+interfaces specified for a particular programming language, one that
+is widely used among developers working in that language.
+
+  The "System Libraries" of an executable work include anything, other
+than the work as a whole, that (a) is included in the normal form of
+packaging a Major Component, but which is not part of that Major
+Component, and (b) serves only to enable use of the work with that
+Major Component, or to implement a Standard Interface for which an
+implementation is available to the public in source code form.  A
+"Major Component", in this context, means a major essential component
+(kernel, window system, and so on) of the specific operating system
+(if any) on which the executable work runs, or a compiler used to
+produce the work, or an object code interpreter used to run it.
+
+  The "Corresponding Source" for a work in object code form means all
+the source code needed to generate, install, and (for an executable
+work) run the object code and to modify the work, including scripts to
+control those activities.  However, it does not include the work's
+System Libraries, or general-purpose tools or generally available free
+programs which are used unmodified in performing those activities but
+which are not part of the work.  For example, Corresponding Source
+includes interface definition files associated with source files for
+the work, and the source code for shared libraries and dynamically
+linked subprograms that the work is specifically designed to require,
+such as by intimate data communication or control flow between those
+subprograms and other parts of the work.
+
+  The Corresponding Source need not include anything that users
+can regenerate automatically from other parts of the Corresponding
+Source.
+
+  The Corresponding Source for a work in source code form is that
+same work.
+
+  2. Basic Permissions.
+
+  All rights granted under this License are granted for the term of
+copyright on the Program, and are irrevocable provided the stated
+conditions are met.  This License explicitly affirms your unlimited
+permission to run the unmodified Program.  The output from running a
+covered work is covered by this License only if the output, given its
+content, constitutes a covered work.  This License acknowledges your
+rights of fair use or other equivalent, as provided by copyright law.
+
+  You may make, run and propagate covered works that you do not
+convey, without conditions so long as your license otherwise remains
+in force.  You may convey covered works to others for the sole purpose
+of having them make modifications exclusively for you, or provide you
+with facilities for running those works, provided that you comply with
+the terms of this License in conveying all material for which you do
+not control copyright.  Those thus making or running the covered works
+for you must do so exclusively on your behalf, under your direction
+and control, on terms that prohibit them from making any copies of
+your copyrighted material outside their relationship with you.
+
+  Conveying under any other circumstances is permitted solely under
+the conditions stated below.  Sublicensing is not allowed; section 10
+makes it unnecessary.
+
+  3. Protecting Users' Legal Rights From Anti-Circumvention Law.
+
+  No covered work shall be deemed part of an effective technological
+measure under any applicable law fulfilling obligations under article
+11 of the WIPO copyright treaty adopted on 20 December 1996, or
+similar laws prohibiting or restricting circumvention of such
+measures.
+
+  When you convey a covered work, you waive any legal power to forbid
+circumvention of technological measures to the extent such circumvention
+is effected by exercising rights under this License with respect to
+the covered work, and you disclaim any intention to limit operation or
+modification of the work as a means of enforcing, against the work's
+users, your or third parties' legal rights to forbid circumvention of
+technological measures.
+
+  4. Conveying Verbatim Copies.
+
+  You may convey verbatim copies of the Program's source code as you
+receive it, in any medium, provided that you conspicuously and
+appropriately publish on each copy an appropriate copyright notice;
+keep intact all notices stating that this License and any
+non-permissive terms added in accord with section 7 apply to the code;
+keep intact all notices of the absence of any warranty; and give all
+recipients a copy of this License along with the Program.
+
+  You may charge any price or no price for each copy that you convey,
+and you may offer support or warranty protection for a fee.
+
+  5. Conveying Modified Source Versions.
+
+  You may convey a work based on the Program, or the modifications to
+produce it from the Program, in the form of source code under the
+terms of section 4, provided that you also meet all of these conditions:
+
+    a) The work must carry prominent notices stating that you modified
+    it, and giving a relevant date.
+
+    b) The work must carry prominent notices stating that it is
+    released under this License and any conditions added under section
+    7.  This requirement modifies the requirement in section 4 to
+    "keep intact all notices".
+
+    c) You must license the entire work, as a whole, under this
+    License to anyone who comes into possession of a copy.  This
+    License will therefore apply, along with any applicable section 7
+    additional terms, to the whole of the work, and all its parts,
+    regardless of how they are packaged.  This License gives no
+    permission to license the work in any other way, but it does not
+    invalidate such permission if you have separately received it.
+
+    d) If the work has interactive user interfaces, each must display
+    Appropriate Legal Notices; however, if the Program has interactive
+    interfaces that do not display Appropriate Legal Notices, your
+    work need not make them do so.
+
+  A compilation of a covered work with other separate and independent
+works, which are not by their nature extensions of the covered work,
+and which are not combined with it such as to form a larger program,
+in or on a volume of a storage or distribution medium, is called an
+"aggregate" if the compilation and its resulting copyright are not
+used to limit the access or legal rights of the compilation's users
+beyond what the individual works permit.  Inclusion of a covered work
+in an aggregate does not cause this License to apply to the other
+parts of the aggregate.
+
+  6. Conveying Non-Source Forms.
+
+  You may convey a covered work in object code form under the terms
+of sections 4 and 5, provided that you also convey the
+machine-readable Corresponding Source under the terms of this License,
+in one of these ways:
+
+    a) Convey the object code in, or embodied in, a physical product
+    (including a physical distribution medium), accompanied by the
+    Corresponding Source fixed on a durable physical medium
+    customarily used for software interchange.
+
+    b) Convey the object code in, or embodied in, a physical product
+    (including a physical distribution medium), accompanied by a
+    written offer, valid for at least three years and valid for as
+    long as you offer spare parts or customer support for that product
+    model, to give anyone who possesses the object code either (1) a
+    copy of the Corresponding Source for all the software in the
+    product that is covered by this License, on a durable physical
+    medium customarily used for software interchange, for a price no
+    more than your reasonable cost of physically performing this
+    conveying of source, or (2) access to copy the
+    Corresponding Source from a network server at no charge.
+
+    c) Convey individual copies of the object code with a copy of the
+    written offer to provide the Corresponding Source.  This
+    alternative is allowed only occasionally and noncommercially, and
+    only if you received the object code with such an offer, in accord
+    with subsection 6b.
+
+    d) Convey the object code by offering access from a designated
+    place (gratis or for a charge), and offer equivalent access to the
+    Corresponding Source in the same way through the same place at no
+    further charge.  You need not require recipients to copy the
+    Corresponding Source along with the object code.  If the place to
+    copy the object code is a network server, the Corresponding Source
+    may be on a different server (operated by you or a third party)
+    that supports equivalent copying facilities, provided you maintain
+    clear directions next to the object code saying where to find the
+    Corresponding Source.  Regardless of what server hosts the
+    Corresponding Source, you remain obligated to ensure that it is
+    available for as long as needed to satisfy these requirements.
+
+    e) Convey the object code using peer-to-peer transmission, provided
+    you inform other peers where the object code and Corresponding
+    Source of the work are being offered to the general public at no
+    charge under subsection 6d.
+
+  A separable portion of the object code, whose source code is excluded
+from the Corresponding Source as a System Library, need not be
+included in conveying the object code work.
+
+  A "User Product" is either (1) a "consumer product", which means any
+tangible personal property which is normally used for personal, family,
+or household purposes, or (2) anything designed or sold for incorporation
+into a dwelling.  In determining whether a product is a consumer product,
+doubtful cases shall be resolved in favor of coverage.  For a particular
+product received by a particular user, "normally used" refers to a
+typical or common use of that class of product, regardless of the status
+of the particular user or of the way in which the particular user
+actually uses, or expects or is expected to use, the product.  A product
+is a consumer product regardless of whether the product has substantial
+commercial, industrial or non-consumer uses, unless such uses represent
+the only significant mode of use of the product.
+
+  "Installation Information" for a User Product means any methods,
+procedures, authorization keys, or other information required to install
+and execute modified versions of a covered work in that User Product from
+a modified version of its Corresponding Source.  The information must
+suffice to ensure that the continued functioning of the modified object
+code is in no case prevented or interfered with solely because
+modification has been made.
+
+  If you convey an object code work under this section in, or with, or
+specifically for use in, a User Product, and the conveying occurs as
+part of a transaction in which the right of possession and use of the
+User Product is transferred to the recipient in perpetuity or for a
+fixed term (regardless of how the transaction is characterized), the
+Corresponding Source conveyed under this section must be accompanied
+by the Installation Information.  But this requirement does not apply
+if neither you nor any third party retains the ability to install
+modified object code on the User Product (for example, the work has
+been installed in ROM).
+
+  The requirement to provide Installation Information does not include a
+requirement to continue to provide support service, warranty, or updates
+for a work that has been modified or installed by the recipient, or for
+the User Product in which it has been modified or installed.  Access to a
+network may be denied when the modification itself materially and
+adversely affects the operation of the network or violates the rules and
+protocols for communication across the network.
+
+  Corresponding Source conveyed, and Installation Information provided,
+in accord with this section must be in a format that is publicly
+documented (and with an implementation available to the public in
+source code form), and must require no special password or key for
+unpacking, reading or copying.
+
+  7. Additional Terms.
+
+  "Additional permissions" are terms that supplement the terms of this
+License by making exceptions from one or more of its conditions.
+Additional permissions that are applicable to the entire Program shall
+be treated as though they were included in this License, to the extent
+that they are valid under applicable law.  If additional permissions
+apply only to part of the Program, that part may be used separately
+under those permissions, but the entire Program remains governed by
+this License without regard to the additional permissions.
+
+  When you convey a copy of a covered work, you may at your option
+remove any additional permissions from that copy, or from any part of
+it.  (Additional permissions may be written to require their own
+removal in certain cases when you modify the work.)  You may place
+additional permissions on material, added by you to a covered work,
+for which you have or can give appropriate copyright permission.
+
+  Notwithstanding any other provision of this License, for material you
+add to a covered work, you may (if authorized by the copyright holders of
+that material) supplement the terms of this License with terms:
+
+    a) Disclaiming warranty or limiting liability differently from the
+    terms of sections 15 and 16 of this License; or
+
+    b) Requiring preservation of specified reasonable legal notices or
+    author attributions in that material or in the Appropriate Legal
+    Notices displayed by works containing it; or
+
+    c) Prohibiting misrepresentation of the origin of that material, or
+    requiring that modified versions of such material be marked in
+    reasonable ways as different from the original version; or
+
+    d) Limiting the use for publicity purposes of names of licensors or
+    authors of the material; or
+
+    e) Declining to grant rights under trademark law for use of some
+    trade names, trademarks, or service marks; or
+
+    f) Requiring indemnification of licensors and authors of that
+    material by anyone who conveys the material (or modified versions of
+    it) with contractual assumptions of liability to the recipient, for
+    any liability that these contractual assumptions directly impose on
+    those licensors and authors.
+
+  All other non-permissive additional terms are considered "further
+restrictions" within the meaning of section 10.  If the Program as you
+received it, or any part of it, contains a notice stating that it is
+governed by this License along with a term that is a further
+restriction, you may remove that term.  If a license document contains
+a further restriction but permits relicensing or conveying under this
+License, you may add to a covered work material governed by the terms
+of that license document, provided that the further restriction does
+not survive such relicensing or conveying.
+
+  If you add terms to a covered work in accord with this section, you
+must place, in the relevant source files, a statement of the
+additional terms that apply to those files, or a notice indicating
+where to find the applicable terms.
+
+  Additional terms, permissive or non-permissive, may be stated in the
+form of a separately written license, or stated as exceptions;
+the above requirements apply either way.
+
+  8. Termination.
+
+  You may not propagate or modify a covered work except as expressly
+provided under this License.  Any attempt otherwise to propagate or
+modify it is void, and will automatically terminate your rights under
+this License (including any patent licenses granted under the third
+paragraph of section 11).
+
+  However, if you cease all violation of this License, then your
+license from a particular copyright holder is reinstated (a)
+provisionally, unless and until the copyright holder explicitly and
+finally terminates your license, and (b) permanently, if the copyright
+holder fails to notify you of the violation by some reasonable means
+prior to 60 days after the cessation.
+
+  Moreover, your license from a particular copyright holder is
+reinstated permanently if the copyright holder notifies you of the
+violation by some reasonable means, this is the first time you have
+received notice of violation of this License (for any work) from that
+copyright holder, and you cure the violation prior to 30 days after
+your receipt of the notice.
+
+  Termination of your rights under this section does not terminate the
+licenses of parties who have received copies or rights from you under
+this License.  If your rights have been terminated and not permanently
+reinstated, you do not qualify to receive new licenses for the same
+material under section 10.
+
+  9. Acceptance Not Required for Having Copies.
+
+  You are not required to accept this License in order to receive or
+run a copy of the Program.  Ancillary propagation of a covered work
+occurring solely as a consequence of using peer-to-peer transmission
+to receive a copy likewise does not require acceptance.  However,
+nothing other than this License grants you permission to propagate or
+modify any covered work.  These actions infringe copyright if you do
+not accept this License.  Therefore, by modifying or propagating a
+covered work, you indicate your acceptance of this License to do so.
+
+  10. Automatic Licensing of Downstream Recipients.
+
+  Each time you convey a covered work, the recipient automatically
+receives a license from the original licensors, to run, modify and
+propagate that work, subject to this License.  You are not responsible
+for enforcing compliance by third parties with this License.
+
+  An "entity transaction" is a transaction transferring control of an
+organization, or substantially all assets of one, or subdividing an
+organization, or merging organizations.  If propagation of a covered
+work results from an entity transaction, each party to that
+transaction who receives a copy of the work also receives whatever
+licenses to the work the party's predecessor in interest had or could
+give under the previous paragraph, plus a right to possession of the
+Corresponding Source of the work from the predecessor in interest, if
+the predecessor has it or can get it with reasonable efforts.
+
+  You may not impose any further restrictions on the exercise of the
+rights granted or affirmed under this License.  For example, you may
+not impose a license fee, royalty, or other charge for exercise of
+rights granted under this License, and you may not initiate litigation
+(including a cross-claim or counterclaim in a lawsuit) alleging that
+any patent claim is infringed by making, using, selling, offering for
+sale, or importing the Program or any portion of it.
+
+  11. Patents.
+
+  A "contributor" is a copyright holder who authorizes use under this
+License of the Program or a work on which the Program is based.  The
+work thus licensed is called the contributor's "contributor version".
+
+  A contributor's "essential patent claims" are all patent claims
+owned or controlled by the contributor, whether already acquired or
+hereafter acquired, that would be infringed by some manner, permitted
+by this License, of making, using, or selling its contributor version,
+but do not include claims that would be infringed only as a
+consequence of further modification of the contributor version.  For
+purposes of this definition, "control" includes the right to grant
+patent sublicenses in a manner consistent with the requirements of
+this License.
+
+  Each contributor grants you a non-exclusive, worldwide, royalty-free
+patent license under the contributor's essential patent claims, to
+make, use, sell, offer for sale, import and otherwise run, modify and
+propagate the contents of its contributor version.
+
+  In the following three paragraphs, a "patent license" is any express
+agreement or commitment, however denominated, not to enforce a patent
+(such as an express permission to practice a patent or covenant not to
+sue for patent infringement).  To "grant" such a patent license to a
+party means to make such an agreement or commitment not to enforce a
+patent against the party.
+
+  If you convey a covered work, knowingly relying on a patent license,
+and the Corresponding Source of the work is not available for anyone
+to copy, free of charge and under the terms of this License, through a
+publicly available network server or other readily accessible means,
+then you must either (1) cause the Corresponding Source to be so
+available, or (2) arrange to deprive yourself of the benefit of the
+patent license for this particular work, or (3) arrange, in a manner
+consistent with the requirements of this License, to extend the patent
+license to downstream recipients.  "Knowingly relying" means you have
+actual knowledge that, but for the patent license, your conveying the
+covered work in a country, or your recipient's use of the covered work
+in a country, would infringe one or more identifiable patents in that
+country that you have reason to believe are valid.
+
+  If, pursuant to or in connection with a single transaction or
+arrangement, you convey, or propagate by procuring conveyance of, a
+covered work, and grant a patent license to some of the parties
+receiving the covered work authorizing them to use, propagate, modify
+or convey a specific copy of the covered work, then the patent license
+you grant is automatically extended to all recipients of the covered
+work and works based on it.
+
+  A patent license is "discriminatory" if it does not include within
+the scope of its coverage, prohibits the exercise of, or is
+conditioned on the non-exercise of one or more of the rights that are
+specifically granted under this License.  You may not convey a covered
+work if you are a party to an arrangement with a third party that is
+in the business of distributing software, under which you make payment
+to the third party based on the extent of your activity of conveying
+the work, and under which the third party grants, to any of the
+parties who would receive the covered work from you, a discriminatory
+patent license (a) in connection with copies of the covered work
+conveyed by you (or copies made from those copies), or (b) primarily
+for and in connection with specific products or compilations that
+contain the covered work, unless you entered into that arrangement,
+or that patent license was granted, prior to 28 March 2007.
+
+  Nothing in this License shall be construed as excluding or limiting
+any implied license or other defenses to infringement that may
+otherwise be available to you under applicable patent law.
+
+  12. No Surrender of Others' Freedom.
+
+  If conditions are imposed on you (whether by court order, agreement or
+otherwise) that contradict the conditions of this License, they do not
+excuse you from the conditions of this License.  If you cannot convey a
+covered work so as to satisfy simultaneously your obligations under this
+License and any other pertinent obligations, then as a consequence you may
+not convey it at all.  For example, if you agree to terms that obligate you
+to collect a royalty for further conveying from those to whom you convey
+the Program, the only way you could satisfy both those terms and this
+License would be to refrain entirely from conveying the Program.
+
+  13. Use with the GNU Affero General Public License.
+
+  Notwithstanding any other provision of this License, you have
+permission to link or combine any covered work with a work licensed
+under version 3 of the GNU Affero General Public License into a single
+combined work, and to convey the resulting work.  The terms of this
+License will continue to apply to the part which is the covered work,
+but the special requirements of the GNU Affero General Public License,
+section 13, concerning interaction through a network will apply to the
+combination as such.
+
+  14. Revised Versions of this License.
+
+  The Free Software Foundation may publish revised and/or new versions of
+the GNU General Public License from time to time.  Such new versions will
+be similar in spirit to the present version, but may differ in detail to
+address new problems or concerns.
+
+  Each version is given a distinguishing version number.  If the
+Program specifies that a certain numbered version of the GNU General
+Public License "or any later version" applies to it, you have the
+option of following the terms and conditions either of that numbered
+version or of any later version published by the Free Software
+Foundation.  If the Program does not specify a version number of the
+GNU General Public License, you may choose any version ever published
+by the Free Software Foundation.
+
+  If the Program specifies that a proxy can decide which future
+versions of the GNU General Public License can be used, that proxy's
+public statement of acceptance of a version permanently authorizes you
+to choose that version for the Program.
+
+  Later license versions may give you additional or different
+permissions.  However, no additional obligations are imposed on any
+author or copyright holder as a result of your choosing to follow a
+later version.
+
+  15. Disclaimer of Warranty.
+
+  THERE IS NO WARRANTY FOR THE PROGRAM, TO THE EXTENT PERMITTED BY
+APPLICABLE LAW.  EXCEPT WHEN OTHERWISE STATED IN WRITING THE COPYRIGHT
+HOLDERS AND/OR OTHER PARTIES PROVIDE THE PROGRAM "AS IS" WITHOUT WARRANTY
+OF ANY KIND, EITHER EXPRESSED OR IMPLIED, INCLUDING, BUT NOT LIMITED TO,
+THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR
+PURPOSE.  THE ENTIRE RISK AS TO THE QUALITY AND PERFORMANCE OF THE PROGRAM
+IS WITH YOU.  SHOULD THE PROGRAM PROVE DEFECTIVE, YOU ASSUME THE COST OF
+ALL NECESSARY SERVICING, REPAIR OR CORRECTION.
+
+  16. Limitation of Liability.
+
+  IN NO EVENT UNLESS REQUIRED BY APPLICABLE LAW OR AGREED TO IN WRITING
+WILL ANY COPYRIGHT HOLDER, OR ANY OTHER PARTY WHO MODIFIES AND/OR CONVEYS
+THE PROGRAM AS PERMITTED ABOVE, BE LIABLE TO YOU FOR DAMAGES, INCLUDING ANY
+GENERAL, SPECIAL, INCIDENTAL OR CONSEQUENTIAL DAMAGES ARISING OUT OF THE
+USE OR INABILITY TO USE THE PROGRAM (INCLUDING BUT NOT LIMITED TO LOSS OF
+DATA OR DATA BEING RENDERED INACCURATE OR LOSSES SUSTAINED BY YOU OR THIRD
+PARTIES OR A FAILURE OF THE PROGRAM TO OPERATE WITH ANY OTHER PROGRAMS),
+EVEN IF SUCH HOLDER OR OTHER PARTY HAS BEEN ADVISED OF THE POSSIBILITY OF
+SUCH DAMAGES.
+
+  17. Interpretation of Sections 15 and 16.
+
+  If the disclaimer of warranty and limitation of liability provided
+above cannot be given local legal effect according to their terms,
+reviewing courts shall apply local law that most closely approximates
+an absolute waiver of all civil liability in connection with the
+Program, unless a warranty or assumption of liability accompanies a
+copy of the Program in return for a fee.
+
+                     END OF TERMS AND CONDITIONS
+
+            How to Apply These Terms to Your New Programs
+
+  If you develop a new program, and you want it to be of the greatest
+possible use to the public, the best way to achieve this is to make it
+free software which everyone can redistribute and change under these terms.
+
+  To do so, attach the following notices to the program.  It is safest
+to attach them to the start of each source file to most effectively
+state the exclusion of warranty; and each file should have at least
+the "copyright" line and a pointer to where the full notice is found.
+
+    <one line to give the program's name and a brief idea of what it does.>
+    Copyright (C) <year>  <name of author>
+
+    This program is free software: you can redistribute it and/or modify
+    it under the terms of the GNU General Public License as published by
+    the Free Software Foundation, either version 3 of the License, or
+    (at your option) any later version.
+
+    This program is distributed in the hope that it will be useful,
+    but WITHOUT ANY WARRANTY; without even the implied warranty of
+    MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
+    GNU General Public License for more details.
+
+    You should have received a copy of the GNU General Public License
+    along with this program.  If not, see <http://www.gnu.org/licenses/>.
+
+Also add information on how to contact you by electronic and paper mail.
+
+  If the program does terminal interaction, make it output a short
+notice like this when it starts in an interactive mode:
+
+    <program>  Copyright (C) <year>  <name of author>
+    This program comes with ABSOLUTELY NO WARRANTY; for details type `show w'.
+    This is free software, and you are welcome to redistribute it
+    under certain conditions; type `show c' for details.
+
+The hypothetical commands `show w' and `show c' should show the appropriate
+parts of the General Public License.  Of course, your program's commands
+might be different; for a GUI interface, you would use an "about box".
+
+  You should also get your employer (if you work as a programmer) or school,
+if any, to sign a "copyright disclaimer" for the program, if necessary.
+For more information on this, and how to apply and follow the GNU GPL, see
+<http://www.gnu.org/licenses/>.
+
+  The GNU General Public License does not permit incorporating your program
+into proprietary programs.  If your program is a subroutine library, you
+may consider it more useful to permit linking proprietary applications with
+the library.  If this is what you want to do, use the GNU Lesser General
+Public License instead of this License.  But first, please read
+<http://www.gnu.org/philosophy/why-not-lgpl.html>.

+ 8 - 0
packages/flutter_ffmpeg_full-gpl/android/.gitignore

@@ -0,0 +1,8 @@
+*.iml
+.gradle
+/local.properties
+/.idea/workspace.xml
+/.idea/libraries
+.DS_Store
+/build
+/captures

+ 38 - 0
packages/flutter_ffmpeg_full-gpl/android/build.gradle

@@ -0,0 +1,38 @@
+group 'com.arthenica.flutter.ffmpeg'
+version '0.1.0'
+
+buildscript {
+    repositories {
+        google()
+        jcenter()
+    }
+
+    dependencies {
+        classpath 'com.android.tools.build:gradle:3.2.1'
+    }
+}
+
+rootProject.allprojects {
+    repositories {
+        google()
+        jcenter()
+    }
+}
+
+apply plugin: 'com.android.library'
+
+android {
+    compileSdkVersion 27
+
+    defaultConfig {
+        minSdkVersion 21
+        testInstrumentationRunner "android.support.test.runner.AndroidJUnitRunner"
+    }
+    lintOptions {
+        disable 'InvalidPackage'
+    }
+}
+
+dependencies {
+    implementation 'com.arthenica:mobile-ffmpeg-full-gpl:4.2.LTS'
+}

+ 1 - 0
packages/flutter_ffmpeg_full-gpl/android/gradle.properties

@@ -0,0 +1 @@
+org.gradle.jvmargs=-Xmx1536M

+ 1 - 0
packages/flutter_ffmpeg_full-gpl/android/settings.gradle

@@ -0,0 +1 @@
+rootProject.name = 'flutter_ffmpeg'

+ 3 - 0
packages/flutter_ffmpeg_full-gpl/android/src/main/AndroidManifest.xml

@@ -0,0 +1,3 @@
+<manifest xmlns:android="http://schemas.android.com/apk/res/android"
+  package="com.arthenica.flutter.ffmpeg">
+</manifest>

+ 66 - 0
packages/flutter_ffmpeg_full-gpl/android/src/main/java/com/arthenica/flutter/ffmpeg/FlutterFFmpegExecuteAsyncArgumentsTask.java

@@ -0,0 +1,66 @@
+/*
+ * Copyright (c) 2019 Taner Sener
+ *
+ * This file is part of FlutterFFmpeg.
+ *
+ * FlutterFFmpeg is free software: you can redistribute it and/or modify
+ * it under the terms of the GNU Lesser General Public License as published by
+ * the Free Software Foundation, either version 3 of the License, or
+ * (at your option) any later version.
+ *
+ * FlutterFFmpeg is distributed in the hope that it will be useful,
+ * but WITHOUT ANY WARRANTY; without even the implied warranty of
+ * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
+ * GNU Lesser General Public License for more details.
+ *
+ * You should have received a copy of the GNU Lesser General Public License
+ * along with FlutterFFmpeg.  If not, see <http://www.gnu.org/licenses/>.
+ */
+
+package com.arthenica.flutter.ffmpeg;
+
+import android.os.AsyncTask;
+import android.util.Log;
+
+import com.arthenica.mobileffmpeg.FFmpeg;
+
+import java.util.Arrays;
+import java.util.List;
+
+import io.flutter.plugin.common.MethodChannel;
+
+/**
+ * Asynchronous task which performs {@link FFmpeg#execute(String[])} method invocations.
+ *
+ * @author Taner Sener
+ * @since 0.1.0
+ */
+public class FlutterFFmpegExecuteAsyncArgumentsTask extends AsyncTask<String, Integer, Integer> {
+
+    private final MethodChannel.Result result;
+    private final List<String> arguments;
+
+    FlutterFFmpegExecuteAsyncArgumentsTask(final List<String> arguments, final MethodChannel.Result result) {
+        this.arguments = arguments;
+        this.result = result;
+    }
+
+    @Override
+    protected Integer doInBackground(final String... dummyString) {
+        final String[] argumentsArray = arguments.toArray(new String[0]);
+
+        Log.d(FlutterFFmpegPlugin.LIBRARY_NAME, String.format("Running FFmpeg with arguments: %s.", Arrays.toString(argumentsArray)));
+
+        int rc = FFmpeg.execute(argumentsArray);
+
+        Log.d(FlutterFFmpegPlugin.LIBRARY_NAME, String.format("FFmpeg exited with rc: %d", rc));
+
+        return rc;
+    }
+
+    @Override
+    protected void onPostExecute(final Integer rc) {
+        result.success(FlutterFFmpegPlugin.toIntMap(FlutterFFmpegPlugin.KEY_RC, rc));
+    }
+
+}

+ 72 - 0
packages/flutter_ffmpeg_full-gpl/android/src/main/java/com/arthenica/flutter/ffmpeg/FlutterFFmpegExecuteAsyncCommandTask.java

@@ -0,0 +1,72 @@
+/*
+ * Copyright (c) 2019 Taner Sener
+ *
+ * This file is part of FlutterFFmpeg.
+ *
+ * FlutterFFmpeg is free software: you can redistribute it and/or modify
+ * it under the terms of the GNU Lesser General Public License as published by
+ * the Free Software Foundation, either version 3 of the License, or
+ * (at your option) any later version.
+ *
+ * FlutterFFmpeg is distributed in the hope that it will be useful,
+ * but WITHOUT ANY WARRANTY; without even the implied warranty of
+ * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
+ * GNU Lesser General Public License for more details.
+ *
+ * You should have received a copy of the GNU Lesser General Public License
+ * along with FlutterFFmpeg.  If not, see <http://www.gnu.org/licenses/>.
+ */
+
+package com.arthenica.flutter.ffmpeg;
+
+import android.os.AsyncTask;
+import android.util.Log;
+
+import com.arthenica.mobileffmpeg.FFmpeg;
+
+import io.flutter.plugin.common.MethodChannel;
+
+/**
+ * Asynchronous task which performs {@link FFmpeg#execute(String, String)} method invocations.
+ *
+ * @author Taner Sener
+ * @since 0.1.0
+ */
+public class FlutterFFmpegExecuteAsyncCommandTask extends AsyncTask<String, Integer, Integer> {
+
+    private String delimiter;
+    private final MethodChannel.Result result;
+
+    FlutterFFmpegExecuteAsyncCommandTask(final String delimiter, final MethodChannel.Result result) {
+        if (delimiter == null) {
+            this.delimiter = " ";
+        } else {
+            this.delimiter = delimiter;
+        }
+
+        this.result = result;
+    }
+
+    @Override
+    protected Integer doInBackground(final String... strings) {
+        int rc = -1;
+
+        if ((strings != null) && (strings.length > 0)) {
+            final String command = strings[0];
+
+            Log.d(FlutterFFmpegPlugin.LIBRARY_NAME, String.format("Running FFmpeg command: %s with delimiter %s.", command, delimiter));
+
+            rc = FFmpeg.execute(command, delimiter);
+
+            Log.d(FlutterFFmpegPlugin.LIBRARY_NAME, String.format("FFmpeg exited with rc: %d", rc));
+        }
+
+        return rc;
+    }
+
+    @Override
+    protected void onPostExecute(final Integer rc) {
+        result.success(FlutterFFmpegPlugin.toIntMap(FlutterFFmpegPlugin.KEY_RC, rc));
+    }
+
+}

+ 70 - 0
packages/flutter_ffmpeg_full-gpl/android/src/main/java/com/arthenica/flutter/ffmpeg/FlutterFFmpegGetMediaInformationAsyncTask.java

@@ -0,0 +1,70 @@
+/*
+ * Copyright (c) 2019 Taner Sener
+ *
+ * This file is part of FlutterFFmpeg.
+ *
+ * FlutterFFmpeg is free software: you can redistribute it and/or modify
+ * it under the terms of the GNU Lesser General Public License as published by
+ * the Free Software Foundation, either version 3 of the License, or
+ * (at your option) any later version.
+ *
+ * FlutterFFmpeg is distributed in the hope that it will be useful,
+ * but WITHOUT ANY WARRANTY; without even the implied warranty of
+ * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
+ * GNU Lesser General Public License for more details.
+ *
+ * You should have received a copy of the GNU Lesser General Public License
+ * along with FlutterFFmpeg.  If not, see <http://www.gnu.org/licenses/>.
+ */
+
+package com.arthenica.flutter.ffmpeg;
+
+import android.os.AsyncTask;
+import android.util.Log;
+
+import com.arthenica.mobileffmpeg.FFmpeg;
+import com.arthenica.mobileffmpeg.MediaInformation;
+
+import io.flutter.plugin.common.MethodChannel;
+
+/**
+ * Asynchronous task which performs {@link FFmpeg#getMediaInformation(String, Long)} method invocations.
+ *
+ * @author Taner Sener
+ * @since 0.1.0
+ */
+public class FlutterFFmpegGetMediaInformationAsyncTask extends AsyncTask<String, Integer, MediaInformation> {
+
+    private Integer timeout;
+    private final MethodChannel.Result result;
+
+    FlutterFFmpegGetMediaInformationAsyncTask(final Integer timeout, final MethodChannel.Result result) {
+        this.timeout = timeout;
+        this.result = result;
+    }
+
+    @Override
+    protected MediaInformation doInBackground(final String... strings) {
+        MediaInformation mediaInformation = null;
+
+        if ((strings != null) && (strings.length > 0)) {
+            final String path = strings[0];
+
+            if (timeout == null) {
+                Log.d(FlutterFFmpegPlugin.LIBRARY_NAME, String.format("Getting media information for %s", path));
+                mediaInformation = FFmpeg.getMediaInformation(path);
+            } else {
+                Log.d(FlutterFFmpegPlugin.LIBRARY_NAME, String.format("Getting media information for %s with timeout %d.", path, timeout.longValue()));
+                mediaInformation = FFmpeg.getMediaInformation(path, timeout.longValue());
+            }
+        }
+
+        return mediaInformation;
+    }
+
+    @Override
+    protected void onPostExecute(final MediaInformation mediaInformation) {
+        result.success(FlutterFFmpegPlugin.toMediaInformationMap(mediaInformation));
+    }
+
+}

+ 436 - 0
packages/flutter_ffmpeg_full-gpl/android/src/main/java/com/arthenica/flutter/ffmpeg/FlutterFFmpegPlugin.java

@@ -0,0 +1,436 @@
+/*
+ * Copyright (c) 2019 Taner Sener
+ *
+ * This file is part of FlutterFFmpeg.
+ *
+ * FlutterFFmpeg is free software: you can redistribute it and/or modify
+ * it under the terms of the GNU Lesser General Public License as published by
+ * the Free Software Foundation, either version 3 of the License, or
+ * (at your option) any later version.
+ *
+ * FlutterFFmpeg is distributed in the hope that it will be useful,
+ * but WITHOUT ANY WARRANTY; without even the implied warranty of
+ * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
+ * GNU Lesser General Public License for more details.
+ *
+ * You should have received a copy of the GNU Lesser General Public License
+ * along with FlutterFFmpeg.  If not, see <http://www.gnu.org/licenses/>.
+ */
+
+package com.arthenica.flutter.ffmpeg;
+
+import android.content.Context;
+import android.system.ErrnoException;
+import android.util.Log;
+
+import com.arthenica.mobileffmpeg.AbiDetect;
+import com.arthenica.mobileffmpeg.Config;
+import com.arthenica.mobileffmpeg.FFmpeg;
+import com.arthenica.mobileffmpeg.Level;
+import com.arthenica.mobileffmpeg.LogCallback;
+import com.arthenica.mobileffmpeg.LogMessage;
+import com.arthenica.mobileffmpeg.MediaInformation;
+import com.arthenica.mobileffmpeg.Statistics;
+import com.arthenica.mobileffmpeg.StatisticsCallback;
+import com.arthenica.mobileffmpeg.StreamInformation;
+
+import java.util.ArrayList;
+import java.util.HashMap;
+import java.util.List;
+import java.util.Map;
+import java.util.Set;
+
+import io.flutter.plugin.common.EventChannel;
+import io.flutter.plugin.common.MethodCall;
+import io.flutter.plugin.common.MethodChannel;
+import io.flutter.plugin.common.MethodChannel.MethodCallHandler;
+import io.flutter.plugin.common.MethodChannel.Result;
+import io.flutter.plugin.common.PluginRegistry.Registrar;
+
+/**
+ * <h3>Flutter FFmpeg Plugin</h3>
+ *
+ * @author Taner Sener
+ * @since 0.1.0
+ */
+public class FlutterFFmpegPlugin implements MethodCallHandler, EventChannel.StreamHandler {
+    public static final String LIBRARY_NAME = "flutter-ffmpeg";
+
+    public static final String PLATFORM_NAME = "android";
+    public static final String KEY_VERSION = "version";
+    public static final String KEY_RC = "rc";
+    public static final String KEY_PLATFORM = "platform";
+    public static final String KEY_PACKAGE_NAME = "packageName";
+    public static final String KEY_LAST_RC = "lastRc";
+
+    public static final String KEY_LAST_COMMAND_OUTPUT = "lastCommandOutput";
+    public static final String KEY_LOG_TEXT = "log";
+
+    public static final String KEY_LOG_LEVEL = "level";
+    public static final String KEY_STAT_TIME = "time";
+    public static final String KEY_STAT_SIZE = "size";
+    public static final String KEY_STAT_BITRATE = "bitrate";
+    public static final String KEY_STAT_SPEED = "speed";
+    public static final String KEY_STAT_VIDEO_FRAME_NUMBER = "videoFrameNumber";
+    public static final String KEY_STAT_VIDEO_QUALITY = "videoQuality";
+    public static final String KEY_STAT_VIDEO_FPS = "videoFps";
+
+    public static final String EVENT_LOG = "FlutterFFmpegLogCallback";
+    public static final String EVENT_STAT = "FlutterFFmpegStatisticsCallback";
+
+    private EventChannel.EventSink eventSink;
+    private final Registrar registrar;
+
+    /**
+     * Registers plugin to registry.
+     *
+     * @param registrar receiver of plugin registration
+     */
+    public static void registerWith(final Registrar registrar) {
+        FlutterFFmpegPlugin handler = new FlutterFFmpegPlugin(registrar);
+
+        final MethodChannel channel = new MethodChannel(registrar.messenger(), "flutter_ffmpeg");
+        channel.setMethodCallHandler(handler);
+
+        final EventChannel eventChannel = new EventChannel(registrar.messenger(), "flutter_ffmpeg_event");
+        eventChannel.setStreamHandler(handler);
+    }
+
+    private FlutterFFmpegPlugin(Registrar registrar) {
+        this.registrar = registrar;
+    }
+
+    private Context getActiveContext() {
+        return (registrar.activity() != null) ? registrar.activity() : registrar.context();
+    }
+
+    /**
+     * Handles method calls.
+     *
+     * @param call   method call
+     * @param result result callback
+     */
+    @Override
+    public void onMethodCall(final MethodCall call, final Result result) {
+        if (call.method.equals("getPlatform")) {
+
+            final String abi = AbiDetect.getAbi();
+            result.success(toStringMap(KEY_PLATFORM, PLATFORM_NAME + "-" + abi));
+
+        } else if (call.method.equals("getFFmpegVersion")) {
+
+            final String version = FFmpeg.getFFmpegVersion();
+            result.success(toStringMap(KEY_VERSION, version));
+
+        } else if (call.method.equals("executeWithArguments")) {
+
+            List<String> arguments = call.argument("arguments");
+
+            final FlutterFFmpegExecuteAsyncArgumentsTask asyncTask = new FlutterFFmpegExecuteAsyncArgumentsTask(arguments, result);
+            asyncTask.execute("dummy-trigger");
+
+        } else if (call.method.equals("execute")) {
+
+            String command = call.argument("command");
+            String delimiter = call.argument("delimiter");
+
+            final FlutterFFmpegExecuteAsyncCommandTask asyncTask = new FlutterFFmpegExecuteAsyncCommandTask(delimiter, result);
+            asyncTask.execute(command);
+
+        } else if (call.method.equals("cancel")) {
+
+            FFmpeg.cancel();
+
+        } else if (call.method.equals("enableRedirection")) {
+
+            Config.enableRedirection();
+
+        } else if (call.method.equals("disableRedirection")) {
+
+            Config.disableRedirection();
+
+        } else if (call.method.equals("getLogLevel")) {
+
+            final Level level = Config.getLogLevel();
+            result.success(toIntMap(KEY_LOG_LEVEL, levelToInt(level)));
+
+        } else if (call.method.equals("setLogLevel")) {
+
+            Integer level = call.argument("level");
+            if (level == null) {
+                level = Level.AV_LOG_TRACE.getValue();
+            }
+            Config.setLogLevel(Level.from(level));
+
+        } else if (call.method.equals("enableLogs")) {
+
+            Config.enableLogCallback(new LogCallback() {
+
+                @Override
+                public void apply(final LogMessage logMessage) {
+                    emitLogMessage(logMessage);
+                }
+            });
+
+        } else if (call.method.equals("disableLogs")) {
+
+            Config.enableLogCallback(null);
+
+        } else if (call.method.equals("enableStatistics")) {
+
+            Config.enableStatisticsCallback(new StatisticsCallback() {
+
+                @Override
+                public void apply(final Statistics statistics) {
+                    emitStatistics(statistics);
+                }
+            });
+
+        } else if (call.method.equals("disableStatistics")) {
+
+            Config.enableStatisticsCallback(null);
+
+        } else if (call.method.equals("getLastReceivedStatistics")) {
+
+            result.success(toMap(Config.getLastReceivedStatistics()));
+
+        } else if (call.method.equals("resetStatistics")) {
+
+            Config.resetStatistics();
+
+        } else if (call.method.equals("setFontconfigConfigurationPath")) {
+            String path = call.argument("path");
+
+            try {
+                Config.setFontconfigConfigurationPath(path);
+            } catch (final ErrnoException e) {
+                Log.w(LIBRARY_NAME, String.format("Setting fontconfig configuration path failed for %s", path), e);
+            }
+
+        } else if (call.method.equals("setFontDirectory")) {
+
+            String path = call.argument("fontDirectory");
+            Map<String, String> map = call.argument("fontNameMap");
+
+            Config.setFontDirectory(getActiveContext(), path, map);
+
+        } else if (call.method.equals("getPackageName")) {
+
+            final String packageName = Config.getPackageName();
+            result.success(toStringMap(KEY_PACKAGE_NAME, packageName));
+
+        } else if (call.method.equals("getExternalLibraries")) {
+
+            final List<String> externalLibraries = Config.getExternalLibraries();
+            result.success(externalLibraries);
+
+        } else if (call.method.equals("getLastReturnCode")) {
+
+            int lastReturnCode = FFmpeg.getLastReturnCode();
+            result.success(toIntMap(KEY_LAST_RC, lastReturnCode));
+
+        } else if (call.method.equals("getLastCommandOutput")) {
+
+            final String lastCommandOutput = FFmpeg.getLastCommandOutput();
+            result.success(toStringMap(KEY_LAST_COMMAND_OUTPUT, lastCommandOutput));
+
+        } else if (call.method.equals("getMediaInformation")) {
+            final String path = call.argument("path");
+            Integer timeout = call.argument("timeout");
+            if (timeout == null) {
+                timeout = 10000;
+            }
+
+            final FlutterFFmpegGetMediaInformationAsyncTask asyncTask = new FlutterFFmpegGetMediaInformationAsyncTask(timeout, result);
+            asyncTask.execute(path);
+
+        } else {
+            result.notImplemented();
+        }
+    }
+
+    @Override
+    public void onListen(Object o, EventChannel.EventSink eventSink) {
+        this.eventSink = eventSink;
+    }
+
+    @Override
+    public void onCancel(Object o) {
+        this.eventSink = null;
+    }
+
+    protected void emitLogMessage(final LogMessage logMessage) {
+        final HashMap<String, Object> logWrapperMap = new HashMap<>();
+        final HashMap<String, Object> logMap = new HashMap<>();
+
+        logMap.put(KEY_LOG_LEVEL, levelToInt(logMessage.getLevel()));
+        logMap.put(KEY_LOG_TEXT, logMessage.getText());
+
+        logWrapperMap.put(EVENT_LOG, logMap);
+
+        eventSink.success(logWrapperMap);
+    }
+
+    protected void emitStatistics(final Statistics statistics) {
+        final HashMap<String, Object> statisticsMap = new HashMap<>();
+        statisticsMap.put(EVENT_STAT, toMap(statistics));
+        eventSink.success(statisticsMap);
+    }
+
+    public static int levelToInt(final Level level) {
+        return (level == null) ? Level.AV_LOG_TRACE.getValue() : level.getValue();
+    }
+
+    public static HashMap<String, String> toStringMap(final String key, final String value) {
+        final HashMap<String, String> map = new HashMap<>();
+        map.put(key, value);
+        return map;
+    }
+
+    public static HashMap<String, Integer> toIntMap(final String key, final int value) {
+        final HashMap<String, Integer> map = new HashMap<>();
+        map.put(key, value);
+        return map;
+    }
+
+    public static Map<String, Object> toMap(final Statistics statistics) {
+        final HashMap<String, Object> statisticsMap = new HashMap<>();
+
+        if (statistics != null) {
+            statisticsMap.put(KEY_STAT_TIME, statistics.getTime());
+            statisticsMap.put(KEY_STAT_SIZE, (statistics.getSize() < Integer.MAX_VALUE) ? (int) statistics.getSize() : (int) (statistics.getSize() % Integer.MAX_VALUE));
+            statisticsMap.put(KEY_STAT_BITRATE, statistics.getBitrate());
+            statisticsMap.put(KEY_STAT_SPEED, statistics.getSpeed());
+
+            statisticsMap.put(KEY_STAT_VIDEO_FRAME_NUMBER, statistics.getVideoFrameNumber());
+            statisticsMap.put(KEY_STAT_VIDEO_QUALITY, statistics.getVideoQuality());
+            statisticsMap.put(KEY_STAT_VIDEO_FPS, statistics.getVideoFps());
+        }
+
+        return statisticsMap;
+    }
+
+    public static HashMap<String, Object> toMediaInformationMap(final MediaInformation mediaInformation) {
+        final HashMap<String, Object> map = new HashMap<>();
+
+        if (mediaInformation != null) {
+            if (mediaInformation.getFormat() != null) {
+                map.put("format", mediaInformation.getFormat());
+            }
+            if (mediaInformation.getPath() != null) {
+                map.put("path", mediaInformation.getPath());
+            }
+            if (mediaInformation.getStartTime() != null) {
+                map.put("startTime", mediaInformation.getStartTime().intValue());
+            }
+            if (mediaInformation.getDuration() != null) {
+                map.put("duration", mediaInformation.getDuration().intValue());
+            }
+            if (mediaInformation.getBitrate() != null) {
+                map.put("bitrate", mediaInformation.getBitrate().intValue());
+            }
+            if (mediaInformation.getRawInformation() != null) {
+                map.put("rawInformation", mediaInformation.getRawInformation());
+            }
+
+            final Set<Map.Entry<String, String>> metadata = mediaInformation.getMetadataEntries();
+            if ((metadata != null) && (metadata.size() > 0)) {
+                final HashMap<String, String> metadataMap = new HashMap<>();
+
+                for (Map.Entry<String, String> entry : metadata) {
+                    metadataMap.put(entry.getKey(), entry.getValue());
+                }
+
+                map.put("metadata", metadataMap);
+            }
+
+            final List<StreamInformation> streams = mediaInformation.getStreams();
+            if ((streams != null) && (streams.size() > 0)) {
+                final ArrayList<Map<String, Object>> array = new ArrayList<>();
+
+                for (StreamInformation streamInformation : streams) {
+                    array.add(toStreamInformationMap(streamInformation));
+                }
+
+                map.put("streams", array);
+            }
+        }
+
+        return map;
+    }
+
+    public static Map<String, Object> toStreamInformationMap(final StreamInformation streamInformation) {
+        final HashMap<String, Object> map = new HashMap<>();
+
+        if (streamInformation != null) {
+            if (streamInformation.getIndex() != null) {
+                map.put("index", streamInformation.getIndex().intValue());
+            }
+            if (streamInformation.getType() != null) {
+                map.put("type", streamInformation.getType());
+            }
+            if (streamInformation.getCodec() != null) {
+                map.put("codec", streamInformation.getCodec());
+            }
+            if (streamInformation.getFullCodec() != null) {
+                map.put("fullCodec", streamInformation.getFullCodec());
+            }
+            if (streamInformation.getFormat() != null) {
+                map.put("format", streamInformation.getFormat());
+            }
+            if (streamInformation.getFullFormat() != null) {
+                map.put("fullFormat", streamInformation.getFullFormat());
+            }
+            if (streamInformation.getWidth() != null) {
+                map.put("width", streamInformation.getWidth().intValue());
+            }
+            if (streamInformation.getHeight() != null) {
+                map.put("height", streamInformation.getHeight().intValue());
+            }
+            if (streamInformation.getBitrate() != null) {
+                map.put("bitrate", streamInformation.getBitrate().intValue());
+            }
+            if (streamInformation.getSampleRate() != null) {
+                map.put("sampleRate", streamInformation.getSampleRate().intValue());
+            }
+            if (streamInformation.getSampleFormat() != null) {
+                map.put("sampleFormat", streamInformation.getSampleFormat());
+            }
+            if (streamInformation.getChannelLayout() != null) {
+                map.put("channelLayout", streamInformation.getChannelLayout());
+            }
+            if (streamInformation.getSampleAspectRatio() != null) {
+                map.put("sampleAspectRatio", streamInformation.getSampleAspectRatio());
+            }
+            if (streamInformation.getDisplayAspectRatio() != null) {
+                map.put("displayAspectRatio", streamInformation.getDisplayAspectRatio());
+            }
+            if (streamInformation.getAverageFrameRate() != null) {
+                map.put("averageFrameRate", streamInformation.getAverageFrameRate());
+            }
+            if (streamInformation.getRealFrameRate() != null) {
+                map.put("realFrameRate", streamInformation.getRealFrameRate());
+            }
+            if (streamInformation.getTimeBase() != null) {
+                map.put("timeBase", streamInformation.getTimeBase());
+            }
+            if (streamInformation.getCodecTimeBase() != null) {
+                map.put("codecTimeBase", streamInformation.getCodecTimeBase());
+            }
+
+            final Set<Map.Entry<String, String>> metadata = streamInformation.getMetadataEntries();
+            if ((metadata != null) && (metadata.size() > 0)) {
+                final HashMap<String, String> metadataMap = new HashMap<>();
+
+                for (Map.Entry<String, String> entry : metadata) {
+                    metadataMap.put(entry.getKey(), entry.getValue());
+                }
+
+                map.put("metadata", metadataMap);
+            }
+        }
+
+        return map;
+    }
+
+}

+ 36 - 0
packages/flutter_ffmpeg_full-gpl/ios/.gitignore

@@ -0,0 +1,36 @@
+.idea/
+.vagrant/
+.sconsign.dblite
+.svn/
+
+.DS_Store
+*.swp
+profile
+
+DerivedData/
+build/
+GeneratedPluginRegistrant.h
+GeneratedPluginRegistrant.m
+
+.generated/
+
+*.pbxuser
+*.mode1v3
+*.mode2v3
+*.perspectivev3
+
+!default.pbxuser
+!default.mode1v3
+!default.mode2v3
+!default.perspectivev3
+
+xcuserdata
+
+*.moved-aside
+
+*.pyc
+*sync/
+Icon?
+.tags*
+
+/Flutter/Generated.xcconfig

+ 0 - 0
packages/flutter_ffmpeg_full-gpl/ios/Assets/.gitkeep


+ 27 - 0
packages/flutter_ffmpeg_full-gpl/ios/Classes/FlutterFFmpegPlugin.h

@@ -0,0 +1,27 @@
+/*
+ * Copyright (c) 2019 Taner Sener
+ *
+ * This file is part of FlutterFFmpeg.
+ *
+ * FlutterFFmpeg is free software: you can redistribute it and/or modify
+ * it under the terms of the GNU Lesser General Public License as published by
+ * the Free Software Foundation, either version 3 of the License, or
+ * (at your option) any later version.
+ *
+ * FlutterFFmpeg is distributed in the hope that it will be useful,
+ * but WITHOUT ANY WARRANTY; without even the implied warranty of
+ * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
+ * GNU Lesser General Public License for more details.
+ *
+ * You should have received a copy of the GNU Lesser General Public License
+ * along with FlutterFFmpeg.  If not, see <http://www.gnu.org/licenses/>.
+ */
+
+#import <Flutter/Flutter.h>
+#import <mobileffmpeg/MobileFFmpegConfig.h>
+
+/**
+ * Flutter FFmpeg Plugin
+ */
+@interface FlutterFFmpegPlugin : NSObject<FlutterPlugin,FlutterStreamHandler,LogDelegate,StatisticsDelegate>
+@end

+ 387 - 0
packages/flutter_ffmpeg_full-gpl/ios/Classes/FlutterFfmpegPlugin.m

@@ -0,0 +1,387 @@
+/*
+ * Copyright (c) 2019 Taner Sener
+ *
+ * This file is part of FlutterFFmpeg.
+ *
+ * FlutterFFmpeg is free software: you can redistribute it and/or modify
+ * it under the terms of the GNU Lesser General Public License as published by
+ * the Free Software Foundation, either version 3 of the License, or
+ * (at your option) any later version.
+ *
+ * FlutterFFmpeg is distributed in the hope that it will be useful,
+ * but WITHOUT ANY WARRANTY; without even the implied warranty of
+ * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
+ * GNU Lesser General Public License for more details.
+ *
+ * You should have received a copy of the GNU Lesser General Public License
+ * along with FlutterFFmpeg.  If not, see <http://www.gnu.org/licenses/>.
+ */
+
+#import "FlutterFFmpegPlugin.h"
+
+#import <mobileffmpeg/ArchDetect.h>
+#import <mobileffmpeg/MobileFFmpeg.h>
+#import <mobileffmpeg/MobileFFmpegConfig.h>
+
+static NSString *const PLATFORM_NAME = @"ios";
+
+static NSString *const KEY_VERSION = @"version";
+static NSString *const KEY_RC = @"rc";
+static NSString *const KEY_PLATFORM = @"platform";
+static NSString *const KEY_PACKAGE_NAME = @"packageName";
+static NSString *const KEY_LAST_RC = @"lastRc";
+static NSString *const KEY_LAST_COMMAND_OUTPUT = @"lastCommandOutput";
+
+static NSString *const KEY_LOG_TEXT = @"log";
+static NSString *const KEY_LOG_LEVEL = @"level";
+
+static NSString *const KEY_STAT_TIME = @"time";
+static NSString *const KEY_STAT_SIZE = @"size";
+static NSString *const KEY_STAT_BITRATE = @"bitrate";
+static NSString *const KEY_STAT_SPEED = @"speed";
+static NSString *const KEY_STAT_VIDEO_FRAME_NUMBER = @"videoFrameNumber";
+static NSString *const KEY_STAT_VIDEO_QUALITY = @"videoQuality";
+static NSString *const KEY_STAT_VIDEO_FPS = @"videoFps";
+
+static NSString *const EVENT_LOG = @"FlutterFFmpegLogCallback";
+static NSString *const EVENT_STAT = @"FlutterFFmpegStatisticsCallback";
+
+/**
+ * Flutter FFmpeg Plugin
+ */
+@implementation FlutterFFmpegPlugin {
+    FlutterEventSink _eventSink;
+}
+
+- (FlutterError *)onListenWithArguments:(id)arguments eventSink:(FlutterEventSink)eventSink {
+    _eventSink = eventSink;
+    return nil;
+}
+
+- (FlutterError *)onCancelWithArguments:(id)arguments {
+    _eventSink = nil;
+    return nil;
+}
+
++ (void)registerWithRegistrar:(NSObject<FlutterPluginRegistrar>*)registrar {
+    FlutterFFmpegPlugin* instance = [[FlutterFFmpegPlugin alloc] init];
+
+    FlutterMethodChannel* methodChannel = [FlutterMethodChannel methodChannelWithName:@"flutter_ffmpeg" binaryMessenger:[registrar messenger]];
+    [registrar addMethodCallDelegate:instance channel:methodChannel];
+
+    FlutterEventChannel* eventChannel = [FlutterEventChannel eventChannelWithName:@"flutter_ffmpeg_event" binaryMessenger:[registrar messenger]];
+    [eventChannel setStreamHandler:instance];
+}
+
+- (void)handleMethodCall:(FlutterMethodCall*)call result:(FlutterResult)result {
+
+    // ARGUMENTS
+    NSArray* arguments = call.arguments[@"arguments"];
+    NSString* command = call.arguments[@"command"];
+    NSString* delimiter = call.arguments[@"delimiter"];
+
+    if ([@"getPlatform" isEqualToString:call.method]) {
+
+        NSString *architecture = [ArchDetect getArch];
+        result([FlutterFFmpegPlugin toStringDictionary:KEY_PLATFORM :[NSString stringWithFormat:@"%@-%@", PLATFORM_NAME, architecture]]);
+
+    } else if ([@"getFFmpegVersion" isEqualToString:call.method]) {
+
+        result([FlutterFFmpegPlugin toStringDictionary:KEY_VERSION :[MobileFFmpeg getFFmpegVersion]]);
+
+    } else if ([@"executeWithArguments" isEqualToString:call.method]) {
+
+        NSLog(@"Running FFmpeg with arguments: %@.\n", arguments);
+
+        dispatch_async(dispatch_get_global_queue(DISPATCH_QUEUE_PRIORITY_DEFAULT, 0), ^{
+
+            int rc = [MobileFFmpeg executeWithArguments:arguments];
+
+            NSLog(@"FFmpeg exited with rc: %d\n", rc);
+
+            result([FlutterFFmpegPlugin toIntDictionary:KEY_RC :[NSNumber numberWithInt:rc]]);
+        });
+
+    } else if ([@"execute" isEqualToString:call.method]) {
+
+        if (delimiter == nil) {
+            delimiter = @" ";
+        }
+
+        NSLog(@"Running FFmpeg command: %@ with delimiter %@.\n", command, delimiter);
+
+        dispatch_async(dispatch_get_global_queue(DISPATCH_QUEUE_PRIORITY_DEFAULT, 0), ^{
+            int rc = [MobileFFmpeg execute:command delimiter:delimiter];
+
+            NSLog(@"FFmpeg exited with rc: %d\n", rc);
+
+            result([FlutterFFmpegPlugin toIntDictionary:KEY_RC :[NSNumber numberWithInt:rc]]);
+        });
+
+    } else if ([@"cancel" isEqualToString:call.method]) {
+
+        [MobileFFmpeg cancel];
+
+    } else if ([@"enableRedirection" isEqualToString:call.method]) {
+
+        [MobileFFmpegConfig enableRedirection];
+
+    } else if ([@"disableRedirection" isEqualToString:call.method]) {
+
+        [MobileFFmpegConfig disableRedirection];
+
+    } else if ([@"getLogLevel" isEqualToString:call.method]) {
+
+        int logLevel = [MobileFFmpegConfig getLogLevel];
+        result([FlutterFFmpegPlugin toIntDictionary:KEY_LOG_LEVEL :[NSNumber numberWithInt:logLevel]]);
+
+    } else if ([@"setLogLevel" isEqualToString:call.method]) {
+
+        NSNumber* logLevel = call.arguments[@"level"];
+        [MobileFFmpegConfig setLogLevel:[logLevel intValue]];
+
+    } else if ([@"enableLogs" isEqualToString:call.method]) {
+
+        [MobileFFmpegConfig setLogDelegate:self];
+
+    } else if ([@"disableLogs" isEqualToString:call.method]) {
+
+        [MobileFFmpegConfig setLogDelegate:nil];
+
+    } else if ([@"enableStatistics" isEqualToString:call.method]) {
+
+        [MobileFFmpegConfig setStatisticsDelegate:self];
+
+    } else if ([@"disableStatistics" isEqualToString:call.method]) {
+
+        [MobileFFmpegConfig setStatisticsDelegate:nil];
+
+    } else if ([@"getLastReceivedStatistics" isEqualToString:call.method]) {
+
+        Statistics *statistics = [MobileFFmpegConfig getLastReceivedStatistics];
+        result([FlutterFFmpegPlugin toStatisticsDictionary:statistics]);
+
+    } else if ([@"resetStatistics" isEqualToString:call.method]) {
+
+        [MobileFFmpegConfig resetStatistics];
+
+    } else if ([@"setFontconfigConfigurationPath" isEqualToString:call.method]) {
+
+        NSString* path = call.arguments[@"path"];
+        [MobileFFmpegConfig setFontconfigConfigurationPath:path];
+
+    } else if ([@"setFontDirectory" isEqualToString:call.method]) {
+
+        NSString* fontDirectoryPath = call.arguments[@"fontDirectory"];
+        NSDictionary* fontNameMapping = call.arguments[@"fontNameMap"];
+        [MobileFFmpegConfig setFontDirectory:fontDirectoryPath with:fontNameMapping];
+
+    } else if ([@"getPackageName" isEqualToString:call.method]) {
+
+        NSString *packageName = [MobileFFmpegConfig getPackageName];
+        result([FlutterFFmpegPlugin toStringDictionary:KEY_PACKAGE_NAME :packageName]);
+
+    } else if ([@"getExternalLibraries" isEqualToString:call.method]) {
+
+        NSArray *externalLibraries = [MobileFFmpegConfig getExternalLibraries];
+        result(externalLibraries);
+
+    } else if ([@"getLastReturnCode" isEqualToString:call.method]) {
+
+        int lastReturnCode = [MobileFFmpeg getLastReturnCode];
+        result([FlutterFFmpegPlugin toIntDictionary:KEY_LAST_RC :[NSNumber numberWithInt:lastReturnCode]]);
+
+    } else if ([@"getLastCommandOutput" isEqualToString:call.method]) {
+
+        NSString *lastCommandOutput = [MobileFFmpeg getLastCommandOutput];
+        result([FlutterFFmpegPlugin toStringDictionary:KEY_LAST_COMMAND_OUTPUT :lastCommandOutput]);
+
+    } else if ([@"getMediaInformation" isEqualToString:call.method]) {
+
+        NSString* path = call.arguments[@"path"];
+        NSNumber* timeout = call.arguments[@"timeout"];
+
+        NSLog(@"Getting media information for %@ with timeout %d.\n", path, [timeout intValue]);
+
+        dispatch_async(dispatch_get_global_queue(DISPATCH_QUEUE_PRIORITY_DEFAULT, 0), ^{
+            MediaInformation *mediaInformation = [MobileFFmpeg getMediaInformation:path timeout:[timeout intValue]];
+            result([FlutterFFmpegPlugin toMediaInformationDictionary:mediaInformation]);
+        });
+
+    } else {
+
+        result(FlutterMethodNotImplemented);
+
+    }
+}
+
+- (void)logCallback: (int)level :(NSString*)message {
+    dispatch_async(dispatch_get_main_queue(), ^{
+        NSMutableDictionary *dictionary = [[NSMutableDictionary alloc] init];
+        dictionary[KEY_LOG_LEVEL] = [NSNumber numberWithInt:level];
+        dictionary[KEY_LOG_TEXT] = message;
+
+        [self emitLogMessage: dictionary];
+    });
+}
+
+- (void)statisticsCallback:(Statistics *)statistics {
+    dispatch_async(dispatch_get_main_queue(), ^{
+        [self emitStatistics: statistics];
+    });
+}
+
+- (void)emitLogMessage:(NSDictionary*)logMessage{
+    _eventSink([FlutterFFmpegPlugin toStringDictionary:EVENT_LOG :logMessage]);
+}
+
+- (void)emitStatistics:(Statistics*)statistics{
+    NSDictionary *dictionary = [FlutterFFmpegPlugin toStatisticsDictionary:statistics];
+    _eventSink([FlutterFFmpegPlugin toStringDictionary:EVENT_STAT :dictionary]);
+}
+
++ (NSDictionary *)toStringDictionary:(NSString*)key :(NSString*)value {
+    NSMutableDictionary *dictionary = [[NSMutableDictionary alloc] init];
+    dictionary[key] = value;
+
+    return dictionary;
+}
+
++ (NSDictionary *)toIntDictionary:(NSString*)key :(NSNumber*)value {
+    NSMutableDictionary *dictionary = [[NSMutableDictionary alloc] init];
+    dictionary[key] = value;
+
+    return dictionary;
+}
+
++ (NSDictionary *)toStatisticsDictionary:(Statistics*)statistics {
+    NSMutableDictionary *dictionary = [[NSMutableDictionary alloc] init];
+
+    if (statistics != nil) {
+        dictionary[KEY_STAT_TIME] = [NSNumber numberWithInt: [statistics getTime]];
+        dictionary[KEY_STAT_SIZE] = [NSNumber numberWithLong: [statistics getSize]];
+
+        dictionary[KEY_STAT_BITRATE] = [NSNumber numberWithDouble: [statistics getBitrate]];
+        dictionary[KEY_STAT_SPEED] = [NSNumber numberWithDouble: [statistics getSpeed]];
+
+        dictionary[KEY_STAT_VIDEO_FRAME_NUMBER] = [NSNumber numberWithInt: [statistics getVideoFrameNumber]];
+        dictionary[KEY_STAT_VIDEO_QUALITY] = [NSNumber numberWithFloat: [statistics getVideoQuality]];
+        dictionary[KEY_STAT_VIDEO_FPS] = [NSNumber numberWithFloat: [statistics getVideoFps]];
+    }
+
+    return dictionary;
+}
+
++ (NSDictionary *)toMediaInformationDictionary:(MediaInformation*)mediaInformation {
+    NSMutableDictionary *dictionary = [[NSMutableDictionary alloc] init];
+
+    if (mediaInformation != nil) {
+        if ([mediaInformation getFormat] != nil) {
+            dictionary[@"format"] =  [mediaInformation getFormat];
+        }
+        if ([mediaInformation getPath] != nil) {
+            dictionary[@"path"] = [mediaInformation getPath];
+        }
+        if ([mediaInformation getStartTime] != nil) {
+            dictionary[@"startTime"] = [mediaInformation getStartTime];
+        }
+        if ([mediaInformation getDuration] != nil) {
+            dictionary[@"duration"] = [mediaInformation getDuration];
+        }
+        if ([mediaInformation getBitrate] != nil) {
+            dictionary[@"bitrate"] = [mediaInformation getBitrate];
+        }
+        if ([mediaInformation getRawInformation] != nil) {
+            dictionary[@"rawInformation"] = [mediaInformation getRawInformation];
+        }
+
+        NSDictionary *metadataDictionary = [mediaInformation getMetadataEntries];
+        if (metadataDictionary != nil && ([metadataDictionary count] > 0)) {
+            dictionary[@"metadata"] = metadataDictionary;
+        }
+
+        NSArray *streams = [mediaInformation getStreams];
+        if (streams != nil && ([streams count] > 0)) {
+            NSMutableArray *array = [[NSMutableArray alloc] init];
+
+            for (int i=0; i < [streams count]; i++) {
+                StreamInformation *streamInformation= [streams objectAtIndex:i];
+                [array addObject: [FlutterFFmpegPlugin toStreamInformationDictionary:streamInformation]];
+            }
+
+            dictionary[@"streams"] = array;
+        }
+    }
+
+    return dictionary;
+}
+
++ (NSDictionary *)toStreamInformationDictionary:(StreamInformation*)streamInformation {
+    NSMutableDictionary *dictionary = [[NSMutableDictionary alloc] init];
+
+    if (streamInformation != nil) {
+        if ([streamInformation getIndex] != nil) {
+            dictionary[@"index"] = [streamInformation getIndex];
+        }
+        if ([streamInformation getType] != nil) {
+            dictionary[@"type"] = [streamInformation getType];
+        }
+        if ([streamInformation getCodec] != nil) {
+            dictionary[@"codec"] = [streamInformation getCodec];
+        }
+        if ([streamInformation getFullCodec] != nil) {
+            dictionary[@"fullCodec"] = [streamInformation getFullCodec];
+        }
+        if ([streamInformation getFormat] != nil) {
+            dictionary[@"format"] = [streamInformation getFormat];
+        }
+        if ([streamInformation getFullFormat] != nil) {
+            dictionary[@"fullFormat"] = [streamInformation getFullFormat];
+        }
+        if ([streamInformation getWidth] != nil) {
+            dictionary[@"width"] = [streamInformation getWidth];
+        }
+        if ([streamInformation getHeight] != nil) {
+            dictionary[@"height"] = [streamInformation getHeight];
+        }
+        if ([streamInformation getBitrate] != nil) {
+            dictionary[@"bitrate"] = [streamInformation getBitrate];
+        }
+        if ([streamInformation getSampleRate] != nil) {
+            dictionary[@"sampleRate"] = [streamInformation getSampleRate];
+        }
+        if ([streamInformation getSampleFormat] != nil) {
+            dictionary[@"sampleFormat"] = [streamInformation getSampleFormat];
+        }
+        if ([streamInformation getChannelLayout] != nil) {
+            dictionary[@"channelLayout"] = [streamInformation getChannelLayout];
+        }
+        if ([streamInformation getSampleAspectRatio] != nil) {
+            dictionary[@"sampleAspectRatio"] = [streamInformation getSampleAspectRatio];
+        }
+        if ([streamInformation getDisplayAspectRatio] != nil) {
+            dictionary[@"displayAspectRatio"] = [streamInformation getDisplayAspectRatio];
+        }
+        if ([streamInformation getAverageFrameRate] != nil) {
+            dictionary[@"averageFrameRate"] = [streamInformation getAverageFrameRate];
+        }
+        if ([streamInformation getRealFrameRate] != nil) {
+            dictionary[@"realFrameRate"] = [streamInformation getRealFrameRate];
+        }
+        if ([streamInformation getTimeBase] != nil) {
+            dictionary[@"timeBase"] = [streamInformation getTimeBase];
+        }
+        if ([streamInformation getCodecTimeBase] != nil) {
+            dictionary[@"codecTimeBase"] = [streamInformation getCodecTimeBase];
+        }
+
+        NSDictionary *metadataDictionary = [streamInformation getMetadataEntries];
+        if (metadataDictionary != nil && ([metadataDictionary count] > 0)) {
+            dictionary[@"metadata"] = metadataDictionary;
+        }
+    }
+
+    return dictionary;
+}
+
+@end

+ 22 - 0
packages/flutter_ffmpeg_full-gpl/ios/flutter_ffmpeg.podspec

@@ -0,0 +1,22 @@
+Pod::Spec.new do |s|
+  s.name             = 'flutter_ffmpeg'
+  s.version          = '0.1.0'
+  s.summary          = 'FFmpeg plugin for Flutter.'
+  s.description      = 'FFmpeg plugin based on mobile-ffmpeg for Flutter.'
+  s.homepage         = 'https://github.com/tanersener/flutter-ffmpeg'
+
+  s.author           = { 'Taner Sener' => 'tanersener@gmail.com' }
+  s.license          = { :file => '../LICENSE' }
+
+  s.requires_arc     = true
+  s.ios.deployment_target = '9.3'
+
+  s.source              = { :path => '.' }
+  s.source_files        = 'Classes/**/*'
+  s.public_header_files = 'Classes/**/*.h'
+
+  s.dependency      'Flutter'
+  s.dependency      'mobile-ffmpeg-full-gpl', '4.2.LTS'
+
+end
+

+ 352 - 0
packages/flutter_ffmpeg_full-gpl/lib/flutter_ffmpeg.dart

@@ -0,0 +1,352 @@
+/*
+ * Copyright (c) 2019 Taner Sener
+ *
+ * This file is part of FlutterFFmpeg.
+ *
+ * FlutterFFmpeg is free software: you can redistribute it and/or modify
+ * it under the terms of the GNU Lesser General Public License as published by
+ * the Free Software Foundation, either version 3 of the License, or
+ * (at your option) any later version.
+ *
+ * FlutterFFmpeg is distributed in the hope that it will be useful,
+ * but WITHOUT ANY WARRANTY; without even the implied warranty of
+ * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
+ * GNU Lesser General Public License for more details.
+ *
+ * You should have received a copy of the GNU Lesser General Public License
+ * along with FlutterFFmpeg.  If not, see <http://www.gnu.org/licenses/>.
+ */
+
+import 'dart:async';
+
+import 'package:flutter/services.dart';
+
+class FlutterFFmpeg {
+  static const MethodChannel _methodChannel = const MethodChannel('flutter_ffmpeg');
+  static const EventChannel _eventChannel = const EventChannel('flutter_ffmpeg_event');
+
+  Function(int level, String message) logCallback;
+  Function(int time, int size, double bitrate, double speed, int videoFrameNumber, double videoQuality, double videoFps) statisticsCallback;
+
+  FlutterFFmpeg() {
+    logCallback = null;
+    statisticsCallback = null;
+
+    print("Loading flutter-ffmpeg.");
+
+    _eventChannel.receiveBroadcastStream().listen(_onEvent, onError: _onError);
+
+    enableLogs();
+    enableStatistics();
+    enableRedirection();
+
+    getPlatform().then((name) => print("Loaded flutter-ffmpeg-$name."));
+  }
+
+  void _onEvent(Object event) {
+    if (event is Map<dynamic, dynamic>) {
+      final Map<String, dynamic> eventMap = event.cast();
+      final Map<dynamic, dynamic> logEvent = eventMap['FlutterFFmpegLogCallback'];
+      final Map<dynamic, dynamic> statisticsEvent = eventMap['FlutterFFmpegStatisticsCallback'];
+
+      if (logEvent != null) {
+        int level = logEvent['level'];
+        String message = logEvent['log'];
+
+        if (this.logCallback == null) {
+          if (message.length > 0) {
+            // PRINT ALREADY ADDS NEW LINE. SO REMOVE THIS ONE
+            if (message.endsWith('\n')) {
+              print(message.substring(0, message.length - 1));
+            } else {
+              print(message);
+            }
+          }
+        } else {
+          this.logCallback(level, message);
+        }
+      }
+
+      if (statisticsEvent != null) {
+        if (this.statisticsCallback != null) {
+          int time = statisticsEvent['time'];
+          int size = statisticsEvent['size'];
+          double bitrate = _doublePrecision(statisticsEvent['bitrate'], 2);
+          double speed = _doublePrecision(statisticsEvent['speed'], 2);
+          int videoFrameNumber = statisticsEvent['videoFrameNumber'];
+          double videoQuality = _doublePrecision(statisticsEvent['videoQuality'], 2);
+          double videoFps = _doublePrecision(statisticsEvent['videoFps'], 2);
+
+          this.statisticsCallback(time, size, bitrate, speed, videoFrameNumber, videoQuality, videoFps);
+        }
+      }
+    }
+  }
+
+  void _onError(Object error) {
+    print('Event error: $error');
+  }
+
+  double _doublePrecision(double value, int precision) {
+    if (value == null) {
+      return 0;
+    } else {
+      return num.parse(value.toStringAsFixed(precision));
+    }
+  }
+
+  /// Returns FFmpeg version bundled within the library.
+  Future<String> getFFmpegVersion() async {
+    try {
+      final Map<dynamic, dynamic> result = await _methodChannel.invokeMethod('getFFmpegVersion');
+      return result['version'];
+    } on PlatformException catch (e) {
+      print("Plugin error: ${e.message}");
+      return null;
+    }
+  }
+
+  /// Returns platform name where library is loaded.
+  Future<String> getPlatform() async {
+    try {
+      final Map<dynamic, dynamic> result = await _methodChannel.invokeMethod('getPlatform');
+      return result['platform'];
+    } on PlatformException catch (e) {
+      print("Plugin error: ${e.message}");
+      return null;
+    }
+  }
+
+  /// Executes FFmpeg with [commandArguments] provided.
+  Future<int> executeWithArguments(List<String> arguments) async {
+    try {
+      final Map<dynamic, dynamic> result = await _methodChannel.invokeMethod('executeWithArguments', {'arguments': arguments});
+      return result['rc'];
+    } on PlatformException catch (e) {
+      print("Plugin error: ${e.message}");
+      return -1;
+    }
+  }
+
+  /// Executes FFmpeg [command] provided. Command is split into arguments using provided [delimiter].
+  Future<int> execute(String command, [String delimiter = ' ']) async {
+    try {
+      final Map<dynamic, dynamic> result = await _methodChannel.invokeMethod('execute', {'command': command, 'delimiter': delimiter});
+      return result['rc'];
+    } on PlatformException catch (e) {
+      print("Plugin error: ${e.message}");
+      return -1;
+    }
+  }
+
+  /// Cancels an ongoing operation.
+  Future<void> cancel() async {
+    try {
+      await _methodChannel.invokeMethod('cancel');
+    } on PlatformException catch (e) {
+      print("Plugin error: ${e.message}");
+    }
+  }
+
+  /// Enables redirection
+  Future<void> enableRedirection() async {
+    try {
+      await _methodChannel.invokeMethod('enableRedirection');
+    } on PlatformException catch (e) {
+      print("Plugin error: ${e.message}");
+    }
+  }
+
+  /// Disables log and statistics redirection. By default redirection is enabled in constructor.
+  /// When redirection is enabled FFmpeg logs are printed to console and can be routed further to a callback function.
+  /// By disabling redirection, logs are redirected to stderr.
+  /// Statistics redirection behaviour is similar. Statistics are not printed at all if redirection is not enabled.
+  /// If it is enabled then it is possible to define a statistics callback function but if you don't, they are not
+  /// printed anywhere and only saved as codelastReceivedStatistics data which can be polled with
+  /// [getLastReceivedStatistics()].
+  Future<void> disableRedirection() async {
+    try {
+      await _methodChannel.invokeMethod('disableRedirection');
+    } on PlatformException catch (e) {
+      print("Plugin error: ${e.message}");
+    }
+  }
+
+  /// Returns log level.
+  Future<int> getLogLevel() async {
+    try {
+      final Map<dynamic, dynamic> result = await _methodChannel.invokeMethod('getLogLevel');
+      return result['level'];
+    } on PlatformException catch (e) {
+      print("Plugin error: ${e.message}");
+      return -1;
+    }
+  }
+
+  /// Sets log level.
+  Future<void> setLogLevel(int logLevel) async {
+    try {
+      await _methodChannel.invokeMethod('setLogLevel', {'level': logLevel});
+    } on PlatformException catch (e) {
+      print("Plugin error: ${e.message}");
+    }
+  }
+
+  /// Enables log events
+  Future<void> enableLogs() async {
+    try {
+      await _methodChannel.invokeMethod('enableLogs');
+    } on PlatformException catch (e) {
+      print("Plugin error: ${e.message}");
+    }
+  }
+
+  /// Disables log functionality of the library. Logs will not be printed to console and log callback will be disabled.
+  /// Note that log functionality is enabled by default.
+  Future<void> disableLogs() async {
+    try {
+      await _methodChannel.invokeMethod('disableLogs');
+    } on PlatformException catch (e) {
+      print("Plugin error: ${e.message}");
+    }
+  }
+
+  /// Enables statistics events.
+  Future<void> enableStatistics() async {
+    try {
+      await _methodChannel.invokeMethod('enableStatistics');
+    } on PlatformException catch (e) {
+      print("Plugin error: ${e.message}");
+    }
+  }
+
+  /// Disables statistics functionality of the library. Statistics callback will be disabled but the last received
+  /// statistics data will be still available.
+  /// Note that statistics functionality is enabled by default.
+  Future<void> disableStatistics() async {
+    try {
+      await _methodChannel.invokeMethod('disableStatistics');
+    } on PlatformException catch (e) {
+      print("Plugin error: ${e.message}");
+    }
+  }
+
+  /// Sets a callback to redirect FFmpeg logs. [newCallback] is a new log callback function, use null to disable a previously defined callback
+  void enableLogCallback(Function(int level, String message) newCallback) {
+    try {
+      this.logCallback = newCallback;
+    } on PlatformException catch (e) {
+      print("Plugin error: ${e.message}");
+    }
+  }
+
+  /// Sets a callback to redirect FFmpeg statistics. [newCallback] is a new statistics callback function, use null to disable a previously defined callback
+  void enableStatisticsCallback(Function(int time, int size, double bitrate, double speed, int videoFrameNumber, double videoQuality, double videoFps) newCallback) {
+    try {
+      this.statisticsCallback = newCallback;
+    } on PlatformException catch (e) {
+      print("Plugin error: ${e.message}");
+    }
+  }
+
+  /// Returns the last received statistics data stored in bitrate, size, speed, time, videoFps, videoFrameNumber and
+  /// videoQuality fields
+  Future<Map<dynamic, dynamic>> getLastReceivedStatistics() async {
+    try {
+      final Map<dynamic, dynamic> result = await _methodChannel.invokeMethod('getLastReceivedStatistics');
+      return result;
+    } on PlatformException catch (e) {
+      print("Plugin error: ${e.message}");
+      return null;
+    }
+  }
+
+  /// Resets last received statistics. It is recommended to call it before starting a new execution.
+  Future<void> resetStatistics() async {
+    try {
+      await _methodChannel.invokeMethod('resetStatistics');
+    } on PlatformException catch (e) {
+      print("Plugin error: ${e.message}");
+    }
+  }
+
+  /// Sets and overrides fontconfig configuration directory.
+  Future<void> setFontconfigConfigurationPath(String path) async {
+    try {
+      await _methodChannel.invokeMethod('setFontconfigConfigurationPath', {'path': path});
+    } on PlatformException catch (e) {
+      print("Plugin error: ${e.message}");
+    }
+  }
+
+  /// Registers fonts inside the given [fontDirectory], so they are available to use in FFmpeg filters.
+  Future<void> setFontDirectory(String fontDirectory, Map<String, String> fontNameMap) async {
+    var parameters;
+    if (fontNameMap == null) {
+      parameters = {'fontDirectory': fontDirectory};
+    } else {
+      parameters = {'fontDirectory': fontDirectory, 'fontNameMap': fontNameMap};
+    }
+
+    try {
+      await _methodChannel.invokeMethod('setFontDirectory', parameters);
+    } on PlatformException catch (e) {
+      print("Plugin error: ${e.message}");
+    }
+  }
+
+  /// Returns FlutterFFmpeg package name.
+  Future<String> getPackageName() async {
+    try {
+      final Map<dynamic, dynamic> result = await _methodChannel.invokeMethod('getPackageName');
+      return result['packageName'];
+    } on PlatformException catch (e) {
+      print("Plugin error: ${e.message}");
+      return null;
+    }
+  }
+
+  /// Returns supported external libraries.
+  Future<List<dynamic>> getExternalLibraries() async {
+    try {
+      final List<dynamic> result = await _methodChannel.invokeMethod('getExternalLibraries');
+      return result;
+    } on PlatformException catch (e) {
+      print("Plugin error: ${e.message}");
+      return null;
+    }
+  }
+
+  /// Returns return code of last executed command.
+  Future<int> getLastReturnCode() async {
+    try {
+      final Map<dynamic, dynamic> result = await _methodChannel.invokeMethod('getLastReturnCode');
+      return result['lastRc'];
+    } on PlatformException catch (e) {
+      print("Plugin error: ${e.message}");
+      return -1;
+    }
+  }
+
+  /// Returns log output of last executed command. Please note that disabling redirection using
+  /// [disableRedirection()] method also disables this functionality.
+  Future<String> getLastCommandOutput() async {
+    try {
+      final Map<dynamic, dynamic> result = await _methodChannel.invokeMethod('getLastCommandOutput');
+      return result['lastCommandOutput'];
+    } on PlatformException catch (e) {
+      print("Plugin error: ${e.message}");
+      return null;
+    }
+  }
+
+  /// Returns media information for given [path] using optional [timeout]
+  Future<Map<dynamic, dynamic>> getMediaInformation(String path, [int timeout = 10000]) async {
+    try {
+      return await _methodChannel.invokeMethod('getMediaInformation', {'path': path, 'timeout': timeout});
+    } on PlatformException catch (e) {
+      print("Plugin error: ${e.message}");
+      return null;
+    }
+  }
+}

+ 76 - 0
packages/flutter_ffmpeg_full-gpl/lib/log_level.dart

@@ -0,0 +1,76 @@
+/*
+ * Copyright (c) 2019 Taner Sener
+ *
+ * This file is part of FlutterFFmpeg.
+ *
+ * FlutterFFmpeg is free software: you can redistribute it and/or modify
+ * it under the terms of the GNU Lesser General Public License as published by
+ * the Free Software Foundation, either version 3 of the License, or
+ * (at your option) any later version.
+ *
+ * FlutterFFmpeg is distributed in the hope that it will be useful,
+ * but WITHOUT ANY WARRANTY; without even the implied warranty of
+ * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
+ * GNU Lesser General Public License for more details.
+ *
+ * You should have received a copy of the GNU Lesser General Public License
+ * along with FlutterFFmpeg.  If not, see <http://www.gnu.org/licenses/>.
+ */
+
+class LogLevel {
+  /// Print no output.
+  static const int AV_LOG_QUIET = -8;
+
+  /// Something went really wrong and we will crash now.
+  static const int AV_LOG_PANIC = 0;
+
+  /// Something went wrong and recovery is not possible.
+  /// For example, no header was found for a format which depends
+  /// on headers or an illegal combination of parameters is used.
+  static const int AV_LOG_FATAL = 8;
+
+  /// Something went wrong and cannot losslessly be recovered.
+  /// However, not all future data is affected.
+  static const int AV_LOG_ERROR = 16;
+
+  /// Something somehow does not look correct. This may or may not
+  /// lead to problems. An example would be the use of '-vstrict -2'.
+  static const int AV_LOG_WARNING = 24;
+
+  /// int Standard information.
+  static const int AV_LOG_INFO = 32;
+
+  /// Detailed information.
+  static const int AV_LOG_VERBOSE = 40;
+
+  /// Stuff which is only useful for libav* developers.
+  static const int AV_LOG_DEBUG = 48;
+
+  /// Extremely verbose debugging, useful for libav* development.
+  static const int AV_LOG_TRACE = 56;
+
+  /// Returns log level string from int
+  static String levelToString(int level) {
+    switch (level) {
+      case LogLevel.AV_LOG_TRACE:
+        return "TRACE";
+      case LogLevel.AV_LOG_DEBUG:
+        return "DEBUG";
+      case LogLevel.AV_LOG_VERBOSE:
+        return "VERBOSE";
+      case LogLevel.AV_LOG_INFO:
+        return "INFO";
+      case LogLevel.AV_LOG_WARNING:
+        return "WARNING";
+      case LogLevel.AV_LOG_ERROR:
+        return "ERROR";
+      case LogLevel.AV_LOG_FATAL:
+        return "FATAL";
+      case LogLevel.AV_LOG_PANIC:
+        return "PANIC";
+      case LogLevel.AV_LOG_QUIET:
+      default:
+        return "";
+    }
+  }
+}

+ 17 - 0
packages/flutter_ffmpeg_full-gpl/pubspec.yaml

@@ -0,0 +1,17 @@
+name: flutter_ffmpeg
+description: FFmpeg plugin for Flutter.
+version: 0.1.0
+author: Taner Sener <tanersener@gmail.com>
+homepage: https://github.com/tanersener/flutter-ffmpeg
+
+environment:
+  sdk: ">=2.0.0-dev.68.0 <3.0.0"
+
+dependencies:
+  flutter:
+    sdk: flutter
+
+flutter:
+  plugin:
+    androidPackage: com.arthenica.flutter.ffmpeg
+    pluginClass: FlutterFFmpegPlugin

+ 7 - 0
packages/flutter_ffmpeg_full/.gitignore

@@ -0,0 +1,7 @@
+.idea/
+.DS_Store
+.dart_tool/
+.pub/
+build/
+pubspec.lock
+.packages

+ 10 - 0
packages/flutter_ffmpeg_full/.metadata

@@ -0,0 +1,10 @@
+# This file tracks properties of this Flutter project.
+# Used by Flutter tool to assess capabilities and perform upgrades etc.
+#
+# This file should be version controlled and should not be manually edited.
+
+version:
+  revision: 5391447fae6209bb21a89e6a5a6583cac1af9b4b
+  channel: stable
+
+project_type: plugin

+ 165 - 0
packages/flutter_ffmpeg_full/LICENSE

@@ -0,0 +1,165 @@
+                   GNU LESSER GENERAL PUBLIC LICENSE
+                       Version 3, 29 June 2007
+
+ Copyright (C) 2007 Free Software Foundation, Inc. <http://fsf.org/>
+ Everyone is permitted to copy and distribute verbatim copies
+ of this license document, but changing it is not allowed.
+
+
+  This version of the GNU Lesser General Public License incorporates
+the terms and conditions of version 3 of the GNU General Public
+License, supplemented by the additional permissions listed below.
+
+  0. Additional Definitions.
+
+  As used herein, "this License" refers to version 3 of the GNU Lesser
+General Public License, and the "GNU GPL" refers to version 3 of the GNU
+General Public License.
+
+  "The Library" refers to a covered work governed by this License,
+other than an Application or a Combined Work as defined below.
+
+  An "Application" is any work that makes use of an interface provided
+by the Library, but which is not otherwise based on the Library.
+Defining a subclass of a class defined by the Library is deemed a mode
+of using an interface provided by the Library.
+
+  A "Combined Work" is a work produced by combining or linking an
+Application with the Library.  The particular version of the Library
+with which the Combined Work was made is also called the "Linked
+Version".
+
+  The "Minimal Corresponding Source" for a Combined Work means the
+Corresponding Source for the Combined Work, excluding any source code
+for portions of the Combined Work that, considered in isolation, are
+based on the Application, and not on the Linked Version.
+
+  The "Corresponding Application Code" for a Combined Work means the
+object code and/or source code for the Application, including any data
+and utility programs needed for reproducing the Combined Work from the
+Application, but excluding the System Libraries of the Combined Work.
+
+  1. Exception to Section 3 of the GNU GPL.
+
+  You may convey a covered work under sections 3 and 4 of this License
+without being bound by section 3 of the GNU GPL.
+
+  2. Conveying Modified Versions.
+
+  If you modify a copy of the Library, and, in your modifications, a
+facility refers to a function or data to be supplied by an Application
+that uses the facility (other than as an argument passed when the
+facility is invoked), then you may convey a copy of the modified
+version:
+
+   a) under this License, provided that you make a good faith effort to
+   ensure that, in the event an Application does not supply the
+   function or data, the facility still operates, and performs
+   whatever part of its purpose remains meaningful, or
+
+   b) under the GNU GPL, with none of the additional permissions of
+   this License applicable to that copy.
+
+  3. Object Code Incorporating Material from Library Header Files.
+
+  The object code form of an Application may incorporate material from
+a header file that is part of the Library.  You may convey such object
+code under terms of your choice, provided that, if the incorporated
+material is not limited to numerical parameters, data structure
+layouts and accessors, or small macros, inline functions and templates
+(ten or fewer lines in length), you do both of the following:
+
+   a) Give prominent notice with each copy of the object code that the
+   Library is used in it and that the Library and its use are
+   covered by this License.
+
+   b) Accompany the object code with a copy of the GNU GPL and this license
+   document.
+
+  4. Combined Works.
+
+  You may convey a Combined Work under terms of your choice that,
+taken together, effectively do not restrict modification of the
+portions of the Library contained in the Combined Work and reverse
+engineering for debugging such modifications, if you also do each of
+the following:
+
+   a) Give prominent notice with each copy of the Combined Work that
+   the Library is used in it and that the Library and its use are
+   covered by this License.
+
+   b) Accompany the Combined Work with a copy of the GNU GPL and this license
+   document.
+
+   c) For a Combined Work that displays copyright notices during
+   execution, include the copyright notice for the Library among
+   these notices, as well as a reference directing the user to the
+   copies of the GNU GPL and this license document.
+
+   d) Do one of the following:
+
+       0) Convey the Minimal Corresponding Source under the terms of this
+       License, and the Corresponding Application Code in a form
+       suitable for, and under terms that permit, the user to
+       recombine or relink the Application with a modified version of
+       the Linked Version to produce a modified Combined Work, in the
+       manner specified by section 6 of the GNU GPL for conveying
+       Corresponding Source.
+
+       1) Use a suitable shared library mechanism for linking with the
+       Library.  A suitable mechanism is one that (a) uses at run time
+       a copy of the Library already present on the user's computer
+       system, and (b) will operate properly with a modified version
+       of the Library that is interface-compatible with the Linked
+       Version.
+
+   e) Provide Installation Information, but only if you would otherwise
+   be required to provide such information under section 6 of the
+   GNU GPL, and only to the extent that such information is
+   necessary to install and execute a modified version of the
+   Combined Work produced by recombining or relinking the
+   Application with a modified version of the Linked Version. (If
+   you use option 4d0, the Installation Information must accompany
+   the Minimal Corresponding Source and Corresponding Application
+   Code. If you use option 4d1, you must provide the Installation
+   Information in the manner specified by section 6 of the GNU GPL
+   for conveying Corresponding Source.)
+
+  5. Combined Libraries.
+
+  You may place library facilities that are a work based on the
+Library side by side in a single library together with other library
+facilities that are not Applications and are not covered by this
+License, and convey such a combined library under terms of your
+choice, if you do both of the following:
+
+   a) Accompany the combined library with a copy of the same work based
+   on the Library, uncombined with any other library facilities,
+   conveyed under the terms of this License.
+
+   b) Give prominent notice with the combined library that part of it
+   is a work based on the Library, and explaining where to find the
+   accompanying uncombined form of the same work.
+
+  6. Revised Versions of the GNU Lesser General Public License.
+
+  The Free Software Foundation may publish revised and/or new versions
+of the GNU Lesser General Public License from time to time. Such new
+versions will be similar in spirit to the present version, but may
+differ in detail to address new problems or concerns.
+
+  Each version is given a distinguishing version number. If the
+Library as you received it specifies that a certain numbered version
+of the GNU Lesser General Public License "or any later version"
+applies to it, you have the option of following the terms and
+conditions either of that published version or of any later version
+published by the Free Software Foundation. If the Library as you
+received it does not specify a version number of the GNU Lesser
+General Public License, you may choose any version of the GNU Lesser
+General Public License ever published by the Free Software Foundation.
+
+  If the Library as you received it specifies that a proxy can decide
+whether future versions of the GNU Lesser General Public License shall
+apply, that proxy's public statement of acceptance of any version is
+permanent authorization for you to choose that version for the
+Library.

+ 8 - 0
packages/flutter_ffmpeg_full/android/.gitignore

@@ -0,0 +1,8 @@
+*.iml
+.gradle
+/local.properties
+/.idea/workspace.xml
+/.idea/libraries
+.DS_Store
+/build
+/captures

+ 38 - 0
packages/flutter_ffmpeg_full/android/build.gradle

@@ -0,0 +1,38 @@
+group 'com.arthenica.flutter.ffmpeg'
+version '0.1.0'
+
+buildscript {
+    repositories {
+        google()
+        jcenter()
+    }
+
+    dependencies {
+        classpath 'com.android.tools.build:gradle:3.2.1'
+    }
+}
+
+rootProject.allprojects {
+    repositories {
+        google()
+        jcenter()
+    }
+}
+
+apply plugin: 'com.android.library'
+
+android {
+    compileSdkVersion 27
+
+    defaultConfig {
+        minSdkVersion 21
+        testInstrumentationRunner "android.support.test.runner.AndroidJUnitRunner"
+    }
+    lintOptions {
+        disable 'InvalidPackage'
+    }
+}
+
+dependencies {
+    implementation 'com.arthenica:mobile-ffmpeg-full:4.2.LTS'
+}

+ 1 - 0
packages/flutter_ffmpeg_full/android/gradle.properties

@@ -0,0 +1 @@
+org.gradle.jvmargs=-Xmx1536M

+ 1 - 0
packages/flutter_ffmpeg_full/android/settings.gradle

@@ -0,0 +1 @@
+rootProject.name = 'flutter_ffmpeg'

+ 3 - 0
packages/flutter_ffmpeg_full/android/src/main/AndroidManifest.xml

@@ -0,0 +1,3 @@
+<manifest xmlns:android="http://schemas.android.com/apk/res/android"
+  package="com.arthenica.flutter.ffmpeg">
+</manifest>

+ 66 - 0
packages/flutter_ffmpeg_full/android/src/main/java/com/arthenica/flutter/ffmpeg/FlutterFFmpegExecuteAsyncArgumentsTask.java

@@ -0,0 +1,66 @@
+/*
+ * Copyright (c) 2019 Taner Sener
+ *
+ * This file is part of FlutterFFmpeg.
+ *
+ * FlutterFFmpeg is free software: you can redistribute it and/or modify
+ * it under the terms of the GNU Lesser General Public License as published by
+ * the Free Software Foundation, either version 3 of the License, or
+ * (at your option) any later version.
+ *
+ * FlutterFFmpeg is distributed in the hope that it will be useful,
+ * but WITHOUT ANY WARRANTY; without even the implied warranty of
+ * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
+ * GNU Lesser General Public License for more details.
+ *
+ * You should have received a copy of the GNU Lesser General Public License
+ * along with FlutterFFmpeg.  If not, see <http://www.gnu.org/licenses/>.
+ */
+
+package com.arthenica.flutter.ffmpeg;
+
+import android.os.AsyncTask;
+import android.util.Log;
+
+import com.arthenica.mobileffmpeg.FFmpeg;
+
+import java.util.Arrays;
+import java.util.List;
+
+import io.flutter.plugin.common.MethodChannel;
+
+/**
+ * Asynchronous task which performs {@link FFmpeg#execute(String[])} method invocations.
+ *
+ * @author Taner Sener
+ * @since 0.1.0
+ */
+public class FlutterFFmpegExecuteAsyncArgumentsTask extends AsyncTask<String, Integer, Integer> {
+
+    private final MethodChannel.Result result;
+    private final List<String> arguments;
+
+    FlutterFFmpegExecuteAsyncArgumentsTask(final List<String> arguments, final MethodChannel.Result result) {
+        this.arguments = arguments;
+        this.result = result;
+    }
+
+    @Override
+    protected Integer doInBackground(final String... dummyString) {
+        final String[] argumentsArray = arguments.toArray(new String[0]);
+
+        Log.d(FlutterFFmpegPlugin.LIBRARY_NAME, String.format("Running FFmpeg with arguments: %s.", Arrays.toString(argumentsArray)));
+
+        int rc = FFmpeg.execute(argumentsArray);
+
+        Log.d(FlutterFFmpegPlugin.LIBRARY_NAME, String.format("FFmpeg exited with rc: %d", rc));
+
+        return rc;
+    }
+
+    @Override
+    protected void onPostExecute(final Integer rc) {
+        result.success(FlutterFFmpegPlugin.toIntMap(FlutterFFmpegPlugin.KEY_RC, rc));
+    }
+
+}

+ 72 - 0
packages/flutter_ffmpeg_full/android/src/main/java/com/arthenica/flutter/ffmpeg/FlutterFFmpegExecuteAsyncCommandTask.java

@@ -0,0 +1,72 @@
+/*
+ * Copyright (c) 2019 Taner Sener
+ *
+ * This file is part of FlutterFFmpeg.
+ *
+ * FlutterFFmpeg is free software: you can redistribute it and/or modify
+ * it under the terms of the GNU Lesser General Public License as published by
+ * the Free Software Foundation, either version 3 of the License, or
+ * (at your option) any later version.
+ *
+ * FlutterFFmpeg is distributed in the hope that it will be useful,
+ * but WITHOUT ANY WARRANTY; without even the implied warranty of
+ * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
+ * GNU Lesser General Public License for more details.
+ *
+ * You should have received a copy of the GNU Lesser General Public License
+ * along with FlutterFFmpeg.  If not, see <http://www.gnu.org/licenses/>.
+ */
+
+package com.arthenica.flutter.ffmpeg;
+
+import android.os.AsyncTask;
+import android.util.Log;
+
+import com.arthenica.mobileffmpeg.FFmpeg;
+
+import io.flutter.plugin.common.MethodChannel;
+
+/**
+ * Asynchronous task which performs {@link FFmpeg#execute(String, String)} method invocations.
+ *
+ * @author Taner Sener
+ * @since 0.1.0
+ */
+public class FlutterFFmpegExecuteAsyncCommandTask extends AsyncTask<String, Integer, Integer> {
+
+    private String delimiter;
+    private final MethodChannel.Result result;
+
+    FlutterFFmpegExecuteAsyncCommandTask(final String delimiter, final MethodChannel.Result result) {
+        if (delimiter == null) {
+            this.delimiter = " ";
+        } else {
+            this.delimiter = delimiter;
+        }
+
+        this.result = result;
+    }
+
+    @Override
+    protected Integer doInBackground(final String... strings) {
+        int rc = -1;
+
+        if ((strings != null) && (strings.length > 0)) {
+            final String command = strings[0];
+
+            Log.d(FlutterFFmpegPlugin.LIBRARY_NAME, String.format("Running FFmpeg command: %s with delimiter %s.", command, delimiter));
+
+            rc = FFmpeg.execute(command, delimiter);
+
+            Log.d(FlutterFFmpegPlugin.LIBRARY_NAME, String.format("FFmpeg exited with rc: %d", rc));
+        }
+
+        return rc;
+    }
+
+    @Override
+    protected void onPostExecute(final Integer rc) {
+        result.success(FlutterFFmpegPlugin.toIntMap(FlutterFFmpegPlugin.KEY_RC, rc));
+    }
+
+}

+ 70 - 0
packages/flutter_ffmpeg_full/android/src/main/java/com/arthenica/flutter/ffmpeg/FlutterFFmpegGetMediaInformationAsyncTask.java

@@ -0,0 +1,70 @@
+/*
+ * Copyright (c) 2019 Taner Sener
+ *
+ * This file is part of FlutterFFmpeg.
+ *
+ * FlutterFFmpeg is free software: you can redistribute it and/or modify
+ * it under the terms of the GNU Lesser General Public License as published by
+ * the Free Software Foundation, either version 3 of the License, or
+ * (at your option) any later version.
+ *
+ * FlutterFFmpeg is distributed in the hope that it will be useful,
+ * but WITHOUT ANY WARRANTY; without even the implied warranty of
+ * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
+ * GNU Lesser General Public License for more details.
+ *
+ * You should have received a copy of the GNU Lesser General Public License
+ * along with FlutterFFmpeg.  If not, see <http://www.gnu.org/licenses/>.
+ */
+
+package com.arthenica.flutter.ffmpeg;
+
+import android.os.AsyncTask;
+import android.util.Log;
+
+import com.arthenica.mobileffmpeg.FFmpeg;
+import com.arthenica.mobileffmpeg.MediaInformation;
+
+import io.flutter.plugin.common.MethodChannel;
+
+/**
+ * Asynchronous task which performs {@link FFmpeg#getMediaInformation(String, Long)} method invocations.
+ *
+ * @author Taner Sener
+ * @since 0.1.0
+ */
+public class FlutterFFmpegGetMediaInformationAsyncTask extends AsyncTask<String, Integer, MediaInformation> {
+
+    private Integer timeout;
+    private final MethodChannel.Result result;
+
+    FlutterFFmpegGetMediaInformationAsyncTask(final Integer timeout, final MethodChannel.Result result) {
+        this.timeout = timeout;
+        this.result = result;
+    }
+
+    @Override
+    protected MediaInformation doInBackground(final String... strings) {
+        MediaInformation mediaInformation = null;
+
+        if ((strings != null) && (strings.length > 0)) {
+            final String path = strings[0];
+
+            if (timeout == null) {
+                Log.d(FlutterFFmpegPlugin.LIBRARY_NAME, String.format("Getting media information for %s", path));
+                mediaInformation = FFmpeg.getMediaInformation(path);
+            } else {
+                Log.d(FlutterFFmpegPlugin.LIBRARY_NAME, String.format("Getting media information for %s with timeout %d.", path, timeout.longValue()));
+                mediaInformation = FFmpeg.getMediaInformation(path, timeout.longValue());
+            }
+        }
+
+        return mediaInformation;
+    }
+
+    @Override
+    protected void onPostExecute(final MediaInformation mediaInformation) {
+        result.success(FlutterFFmpegPlugin.toMediaInformationMap(mediaInformation));
+    }
+
+}

+ 436 - 0
packages/flutter_ffmpeg_full/android/src/main/java/com/arthenica/flutter/ffmpeg/FlutterFFmpegPlugin.java

@@ -0,0 +1,436 @@
+/*
+ * Copyright (c) 2019 Taner Sener
+ *
+ * This file is part of FlutterFFmpeg.
+ *
+ * FlutterFFmpeg is free software: you can redistribute it and/or modify
+ * it under the terms of the GNU Lesser General Public License as published by
+ * the Free Software Foundation, either version 3 of the License, or
+ * (at your option) any later version.
+ *
+ * FlutterFFmpeg is distributed in the hope that it will be useful,
+ * but WITHOUT ANY WARRANTY; without even the implied warranty of
+ * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
+ * GNU Lesser General Public License for more details.
+ *
+ * You should have received a copy of the GNU Lesser General Public License
+ * along with FlutterFFmpeg.  If not, see <http://www.gnu.org/licenses/>.
+ */
+
+package com.arthenica.flutter.ffmpeg;
+
+import android.content.Context;
+import android.system.ErrnoException;
+import android.util.Log;
+
+import com.arthenica.mobileffmpeg.AbiDetect;
+import com.arthenica.mobileffmpeg.Config;
+import com.arthenica.mobileffmpeg.FFmpeg;
+import com.arthenica.mobileffmpeg.Level;
+import com.arthenica.mobileffmpeg.LogCallback;
+import com.arthenica.mobileffmpeg.LogMessage;
+import com.arthenica.mobileffmpeg.MediaInformation;
+import com.arthenica.mobileffmpeg.Statistics;
+import com.arthenica.mobileffmpeg.StatisticsCallback;
+import com.arthenica.mobileffmpeg.StreamInformation;
+
+import java.util.ArrayList;
+import java.util.HashMap;
+import java.util.List;
+import java.util.Map;
+import java.util.Set;
+
+import io.flutter.plugin.common.EventChannel;
+import io.flutter.plugin.common.MethodCall;
+import io.flutter.plugin.common.MethodChannel;
+import io.flutter.plugin.common.MethodChannel.MethodCallHandler;
+import io.flutter.plugin.common.MethodChannel.Result;
+import io.flutter.plugin.common.PluginRegistry.Registrar;
+
+/**
+ * <h3>Flutter FFmpeg Plugin</h3>
+ *
+ * @author Taner Sener
+ * @since 0.1.0
+ */
+public class FlutterFFmpegPlugin implements MethodCallHandler, EventChannel.StreamHandler {
+    public static final String LIBRARY_NAME = "flutter-ffmpeg";
+
+    public static final String PLATFORM_NAME = "android";
+    public static final String KEY_VERSION = "version";
+    public static final String KEY_RC = "rc";
+    public static final String KEY_PLATFORM = "platform";
+    public static final String KEY_PACKAGE_NAME = "packageName";
+    public static final String KEY_LAST_RC = "lastRc";
+
+    public static final String KEY_LAST_COMMAND_OUTPUT = "lastCommandOutput";
+    public static final String KEY_LOG_TEXT = "log";
+
+    public static final String KEY_LOG_LEVEL = "level";
+    public static final String KEY_STAT_TIME = "time";
+    public static final String KEY_STAT_SIZE = "size";
+    public static final String KEY_STAT_BITRATE = "bitrate";
+    public static final String KEY_STAT_SPEED = "speed";
+    public static final String KEY_STAT_VIDEO_FRAME_NUMBER = "videoFrameNumber";
+    public static final String KEY_STAT_VIDEO_QUALITY = "videoQuality";
+    public static final String KEY_STAT_VIDEO_FPS = "videoFps";
+
+    public static final String EVENT_LOG = "FlutterFFmpegLogCallback";
+    public static final String EVENT_STAT = "FlutterFFmpegStatisticsCallback";
+
+    private EventChannel.EventSink eventSink;
+    private final Registrar registrar;
+
+    /**
+     * Registers plugin to registry.
+     *
+     * @param registrar receiver of plugin registration
+     */
+    public static void registerWith(final Registrar registrar) {
+        FlutterFFmpegPlugin handler = new FlutterFFmpegPlugin(registrar);
+
+        final MethodChannel channel = new MethodChannel(registrar.messenger(), "flutter_ffmpeg");
+        channel.setMethodCallHandler(handler);
+
+        final EventChannel eventChannel = new EventChannel(registrar.messenger(), "flutter_ffmpeg_event");
+        eventChannel.setStreamHandler(handler);
+    }
+
+    private FlutterFFmpegPlugin(Registrar registrar) {
+        this.registrar = registrar;
+    }
+
+    private Context getActiveContext() {
+        return (registrar.activity() != null) ? registrar.activity() : registrar.context();
+    }
+
+    /**
+     * Handles method calls.
+     *
+     * @param call   method call
+     * @param result result callback
+     */
+    @Override
+    public void onMethodCall(final MethodCall call, final Result result) {
+        if (call.method.equals("getPlatform")) {
+
+            final String abi = AbiDetect.getAbi();
+            result.success(toStringMap(KEY_PLATFORM, PLATFORM_NAME + "-" + abi));
+
+        } else if (call.method.equals("getFFmpegVersion")) {
+
+            final String version = FFmpeg.getFFmpegVersion();
+            result.success(toStringMap(KEY_VERSION, version));
+
+        } else if (call.method.equals("executeWithArguments")) {
+
+            List<String> arguments = call.argument("arguments");
+
+            final FlutterFFmpegExecuteAsyncArgumentsTask asyncTask = new FlutterFFmpegExecuteAsyncArgumentsTask(arguments, result);
+            asyncTask.execute("dummy-trigger");
+
+        } else if (call.method.equals("execute")) {
+
+            String command = call.argument("command");
+            String delimiter = call.argument("delimiter");
+
+            final FlutterFFmpegExecuteAsyncCommandTask asyncTask = new FlutterFFmpegExecuteAsyncCommandTask(delimiter, result);
+            asyncTask.execute(command);
+
+        } else if (call.method.equals("cancel")) {
+
+            FFmpeg.cancel();
+
+        } else if (call.method.equals("enableRedirection")) {
+
+            Config.enableRedirection();
+
+        } else if (call.method.equals("disableRedirection")) {
+
+            Config.disableRedirection();
+
+        } else if (call.method.equals("getLogLevel")) {
+
+            final Level level = Config.getLogLevel();
+            result.success(toIntMap(KEY_LOG_LEVEL, levelToInt(level)));
+
+        } else if (call.method.equals("setLogLevel")) {
+
+            Integer level = call.argument("level");
+            if (level == null) {
+                level = Level.AV_LOG_TRACE.getValue();
+            }
+            Config.setLogLevel(Level.from(level));
+
+        } else if (call.method.equals("enableLogs")) {
+
+            Config.enableLogCallback(new LogCallback() {
+
+                @Override
+                public void apply(final LogMessage logMessage) {
+                    emitLogMessage(logMessage);
+                }
+            });
+
+        } else if (call.method.equals("disableLogs")) {
+
+            Config.enableLogCallback(null);
+
+        } else if (call.method.equals("enableStatistics")) {
+
+            Config.enableStatisticsCallback(new StatisticsCallback() {
+
+                @Override
+                public void apply(final Statistics statistics) {
+                    emitStatistics(statistics);
+                }
+            });
+
+        } else if (call.method.equals("disableStatistics")) {
+
+            Config.enableStatisticsCallback(null);
+
+        } else if (call.method.equals("getLastReceivedStatistics")) {
+
+            result.success(toMap(Config.getLastReceivedStatistics()));
+
+        } else if (call.method.equals("resetStatistics")) {
+
+            Config.resetStatistics();
+
+        } else if (call.method.equals("setFontconfigConfigurationPath")) {
+            String path = call.argument("path");
+
+            try {
+                Config.setFontconfigConfigurationPath(path);
+            } catch (final ErrnoException e) {
+                Log.w(LIBRARY_NAME, String.format("Setting fontconfig configuration path failed for %s", path), e);
+            }
+
+        } else if (call.method.equals("setFontDirectory")) {
+
+            String path = call.argument("fontDirectory");
+            Map<String, String> map = call.argument("fontNameMap");
+
+            Config.setFontDirectory(getActiveContext(), path, map);
+
+        } else if (call.method.equals("getPackageName")) {
+
+            final String packageName = Config.getPackageName();
+            result.success(toStringMap(KEY_PACKAGE_NAME, packageName));
+
+        } else if (call.method.equals("getExternalLibraries")) {
+
+            final List<String> externalLibraries = Config.getExternalLibraries();
+            result.success(externalLibraries);
+
+        } else if (call.method.equals("getLastReturnCode")) {
+
+            int lastReturnCode = FFmpeg.getLastReturnCode();
+            result.success(toIntMap(KEY_LAST_RC, lastReturnCode));
+
+        } else if (call.method.equals("getLastCommandOutput")) {
+
+            final String lastCommandOutput = FFmpeg.getLastCommandOutput();
+            result.success(toStringMap(KEY_LAST_COMMAND_OUTPUT, lastCommandOutput));
+
+        } else if (call.method.equals("getMediaInformation")) {
+            final String path = call.argument("path");
+            Integer timeout = call.argument("timeout");
+            if (timeout == null) {
+                timeout = 10000;
+            }
+
+            final FlutterFFmpegGetMediaInformationAsyncTask asyncTask = new FlutterFFmpegGetMediaInformationAsyncTask(timeout, result);
+            asyncTask.execute(path);
+
+        } else {
+            result.notImplemented();
+        }
+    }
+
+    @Override
+    public void onListen(Object o, EventChannel.EventSink eventSink) {
+        this.eventSink = eventSink;
+    }
+
+    @Override
+    public void onCancel(Object o) {
+        this.eventSink = null;
+    }
+
+    protected void emitLogMessage(final LogMessage logMessage) {
+        final HashMap<String, Object> logWrapperMap = new HashMap<>();
+        final HashMap<String, Object> logMap = new HashMap<>();
+
+        logMap.put(KEY_LOG_LEVEL, levelToInt(logMessage.getLevel()));
+        logMap.put(KEY_LOG_TEXT, logMessage.getText());
+
+        logWrapperMap.put(EVENT_LOG, logMap);
+
+        eventSink.success(logWrapperMap);
+    }
+
+    protected void emitStatistics(final Statistics statistics) {
+        final HashMap<String, Object> statisticsMap = new HashMap<>();
+        statisticsMap.put(EVENT_STAT, toMap(statistics));
+        eventSink.success(statisticsMap);
+    }
+
+    public static int levelToInt(final Level level) {
+        return (level == null) ? Level.AV_LOG_TRACE.getValue() : level.getValue();
+    }
+
+    public static HashMap<String, String> toStringMap(final String key, final String value) {
+        final HashMap<String, String> map = new HashMap<>();
+        map.put(key, value);
+        return map;
+    }
+
+    public static HashMap<String, Integer> toIntMap(final String key, final int value) {
+        final HashMap<String, Integer> map = new HashMap<>();
+        map.put(key, value);
+        return map;
+    }
+
+    public static Map<String, Object> toMap(final Statistics statistics) {
+        final HashMap<String, Object> statisticsMap = new HashMap<>();
+
+        if (statistics != null) {
+            statisticsMap.put(KEY_STAT_TIME, statistics.getTime());
+            statisticsMap.put(KEY_STAT_SIZE, (statistics.getSize() < Integer.MAX_VALUE) ? (int) statistics.getSize() : (int) (statistics.getSize() % Integer.MAX_VALUE));
+            statisticsMap.put(KEY_STAT_BITRATE, statistics.getBitrate());
+            statisticsMap.put(KEY_STAT_SPEED, statistics.getSpeed());
+
+            statisticsMap.put(KEY_STAT_VIDEO_FRAME_NUMBER, statistics.getVideoFrameNumber());
+            statisticsMap.put(KEY_STAT_VIDEO_QUALITY, statistics.getVideoQuality());
+            statisticsMap.put(KEY_STAT_VIDEO_FPS, statistics.getVideoFps());
+        }
+
+        return statisticsMap;
+    }
+
+    public static HashMap<String, Object> toMediaInformationMap(final MediaInformation mediaInformation) {
+        final HashMap<String, Object> map = new HashMap<>();
+
+        if (mediaInformation != null) {
+            if (mediaInformation.getFormat() != null) {
+                map.put("format", mediaInformation.getFormat());
+            }
+            if (mediaInformation.getPath() != null) {
+                map.put("path", mediaInformation.getPath());
+            }
+            if (mediaInformation.getStartTime() != null) {
+                map.put("startTime", mediaInformation.getStartTime().intValue());
+            }
+            if (mediaInformation.getDuration() != null) {
+                map.put("duration", mediaInformation.getDuration().intValue());
+            }
+            if (mediaInformation.getBitrate() != null) {
+                map.put("bitrate", mediaInformation.getBitrate().intValue());
+            }
+            if (mediaInformation.getRawInformation() != null) {
+                map.put("rawInformation", mediaInformation.getRawInformation());
+            }
+
+            final Set<Map.Entry<String, String>> metadata = mediaInformation.getMetadataEntries();
+            if ((metadata != null) && (metadata.size() > 0)) {
+                final HashMap<String, String> metadataMap = new HashMap<>();
+
+                for (Map.Entry<String, String> entry : metadata) {
+                    metadataMap.put(entry.getKey(), entry.getValue());
+                }
+
+                map.put("metadata", metadataMap);
+            }
+
+            final List<StreamInformation> streams = mediaInformation.getStreams();
+            if ((streams != null) && (streams.size() > 0)) {
+                final ArrayList<Map<String, Object>> array = new ArrayList<>();
+
+                for (StreamInformation streamInformation : streams) {
+                    array.add(toStreamInformationMap(streamInformation));
+                }
+
+                map.put("streams", array);
+            }
+        }
+
+        return map;
+    }
+
+    public static Map<String, Object> toStreamInformationMap(final StreamInformation streamInformation) {
+        final HashMap<String, Object> map = new HashMap<>();
+
+        if (streamInformation != null) {
+            if (streamInformation.getIndex() != null) {
+                map.put("index", streamInformation.getIndex().intValue());
+            }
+            if (streamInformation.getType() != null) {
+                map.put("type", streamInformation.getType());
+            }
+            if (streamInformation.getCodec() != null) {
+                map.put("codec", streamInformation.getCodec());
+            }
+            if (streamInformation.getFullCodec() != null) {
+                map.put("fullCodec", streamInformation.getFullCodec());
+            }
+            if (streamInformation.getFormat() != null) {
+                map.put("format", streamInformation.getFormat());
+            }
+            if (streamInformation.getFullFormat() != null) {
+                map.put("fullFormat", streamInformation.getFullFormat());
+            }
+            if (streamInformation.getWidth() != null) {
+                map.put("width", streamInformation.getWidth().intValue());
+            }
+            if (streamInformation.getHeight() != null) {
+                map.put("height", streamInformation.getHeight().intValue());
+            }
+            if (streamInformation.getBitrate() != null) {
+                map.put("bitrate", streamInformation.getBitrate().intValue());
+            }
+            if (streamInformation.getSampleRate() != null) {
+                map.put("sampleRate", streamInformation.getSampleRate().intValue());
+            }
+            if (streamInformation.getSampleFormat() != null) {
+                map.put("sampleFormat", streamInformation.getSampleFormat());
+            }
+            if (streamInformation.getChannelLayout() != null) {
+                map.put("channelLayout", streamInformation.getChannelLayout());
+            }
+            if (streamInformation.getSampleAspectRatio() != null) {
+                map.put("sampleAspectRatio", streamInformation.getSampleAspectRatio());
+            }
+            if (streamInformation.getDisplayAspectRatio() != null) {
+                map.put("displayAspectRatio", streamInformation.getDisplayAspectRatio());
+            }
+            if (streamInformation.getAverageFrameRate() != null) {
+                map.put("averageFrameRate", streamInformation.getAverageFrameRate());
+            }
+            if (streamInformation.getRealFrameRate() != null) {
+                map.put("realFrameRate", streamInformation.getRealFrameRate());
+            }
+            if (streamInformation.getTimeBase() != null) {
+                map.put("timeBase", streamInformation.getTimeBase());
+            }
+            if (streamInformation.getCodecTimeBase() != null) {
+                map.put("codecTimeBase", streamInformation.getCodecTimeBase());
+            }
+
+            final Set<Map.Entry<String, String>> metadata = streamInformation.getMetadataEntries();
+            if ((metadata != null) && (metadata.size() > 0)) {
+                final HashMap<String, String> metadataMap = new HashMap<>();
+
+                for (Map.Entry<String, String> entry : metadata) {
+                    metadataMap.put(entry.getKey(), entry.getValue());
+                }
+
+                map.put("metadata", metadataMap);
+            }
+        }
+
+        return map;
+    }
+
+}

+ 36 - 0
packages/flutter_ffmpeg_full/ios/.gitignore

@@ -0,0 +1,36 @@
+.idea/
+.vagrant/
+.sconsign.dblite
+.svn/
+
+.DS_Store
+*.swp
+profile
+
+DerivedData/
+build/
+GeneratedPluginRegistrant.h
+GeneratedPluginRegistrant.m
+
+.generated/
+
+*.pbxuser
+*.mode1v3
+*.mode2v3
+*.perspectivev3
+
+!default.pbxuser
+!default.mode1v3
+!default.mode2v3
+!default.perspectivev3
+
+xcuserdata
+
+*.moved-aside
+
+*.pyc
+*sync/
+Icon?
+.tags*
+
+/Flutter/Generated.xcconfig

+ 0 - 0
packages/flutter_ffmpeg_full/ios/Assets/.gitkeep


+ 27 - 0
packages/flutter_ffmpeg_full/ios/Classes/FlutterFFmpegPlugin.h

@@ -0,0 +1,27 @@
+/*
+ * Copyright (c) 2019 Taner Sener
+ *
+ * This file is part of FlutterFFmpeg.
+ *
+ * FlutterFFmpeg is free software: you can redistribute it and/or modify
+ * it under the terms of the GNU Lesser General Public License as published by
+ * the Free Software Foundation, either version 3 of the License, or
+ * (at your option) any later version.
+ *
+ * FlutterFFmpeg is distributed in the hope that it will be useful,
+ * but WITHOUT ANY WARRANTY; without even the implied warranty of
+ * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
+ * GNU Lesser General Public License for more details.
+ *
+ * You should have received a copy of the GNU Lesser General Public License
+ * along with FlutterFFmpeg.  If not, see <http://www.gnu.org/licenses/>.
+ */
+
+#import <Flutter/Flutter.h>
+#import <mobileffmpeg/MobileFFmpegConfig.h>
+
+/**
+ * Flutter FFmpeg Plugin
+ */
+@interface FlutterFFmpegPlugin : NSObject<FlutterPlugin,FlutterStreamHandler,LogDelegate,StatisticsDelegate>
+@end

+ 387 - 0
packages/flutter_ffmpeg_full/ios/Classes/FlutterFfmpegPlugin.m

@@ -0,0 +1,387 @@
+/*
+ * Copyright (c) 2019 Taner Sener
+ *
+ * This file is part of FlutterFFmpeg.
+ *
+ * FlutterFFmpeg is free software: you can redistribute it and/or modify
+ * it under the terms of the GNU Lesser General Public License as published by
+ * the Free Software Foundation, either version 3 of the License, or
+ * (at your option) any later version.
+ *
+ * FlutterFFmpeg is distributed in the hope that it will be useful,
+ * but WITHOUT ANY WARRANTY; without even the implied warranty of
+ * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
+ * GNU Lesser General Public License for more details.
+ *
+ * You should have received a copy of the GNU Lesser General Public License
+ * along with FlutterFFmpeg.  If not, see <http://www.gnu.org/licenses/>.
+ */
+
+#import "FlutterFFmpegPlugin.h"
+
+#import <mobileffmpeg/ArchDetect.h>
+#import <mobileffmpeg/MobileFFmpeg.h>
+#import <mobileffmpeg/MobileFFmpegConfig.h>
+
+static NSString *const PLATFORM_NAME = @"ios";
+
+static NSString *const KEY_VERSION = @"version";
+static NSString *const KEY_RC = @"rc";
+static NSString *const KEY_PLATFORM = @"platform";
+static NSString *const KEY_PACKAGE_NAME = @"packageName";
+static NSString *const KEY_LAST_RC = @"lastRc";
+static NSString *const KEY_LAST_COMMAND_OUTPUT = @"lastCommandOutput";
+
+static NSString *const KEY_LOG_TEXT = @"log";
+static NSString *const KEY_LOG_LEVEL = @"level";
+
+static NSString *const KEY_STAT_TIME = @"time";
+static NSString *const KEY_STAT_SIZE = @"size";
+static NSString *const KEY_STAT_BITRATE = @"bitrate";
+static NSString *const KEY_STAT_SPEED = @"speed";
+static NSString *const KEY_STAT_VIDEO_FRAME_NUMBER = @"videoFrameNumber";
+static NSString *const KEY_STAT_VIDEO_QUALITY = @"videoQuality";
+static NSString *const KEY_STAT_VIDEO_FPS = @"videoFps";
+
+static NSString *const EVENT_LOG = @"FlutterFFmpegLogCallback";
+static NSString *const EVENT_STAT = @"FlutterFFmpegStatisticsCallback";
+
+/**
+ * Flutter FFmpeg Plugin
+ */
+@implementation FlutterFFmpegPlugin {
+    FlutterEventSink _eventSink;
+}
+
+- (FlutterError *)onListenWithArguments:(id)arguments eventSink:(FlutterEventSink)eventSink {
+    _eventSink = eventSink;
+    return nil;
+}
+
+- (FlutterError *)onCancelWithArguments:(id)arguments {
+    _eventSink = nil;
+    return nil;
+}
+
++ (void)registerWithRegistrar:(NSObject<FlutterPluginRegistrar>*)registrar {
+    FlutterFFmpegPlugin* instance = [[FlutterFFmpegPlugin alloc] init];
+
+    FlutterMethodChannel* methodChannel = [FlutterMethodChannel methodChannelWithName:@"flutter_ffmpeg" binaryMessenger:[registrar messenger]];
+    [registrar addMethodCallDelegate:instance channel:methodChannel];
+
+    FlutterEventChannel* eventChannel = [FlutterEventChannel eventChannelWithName:@"flutter_ffmpeg_event" binaryMessenger:[registrar messenger]];
+    [eventChannel setStreamHandler:instance];
+}
+
+- (void)handleMethodCall:(FlutterMethodCall*)call result:(FlutterResult)result {
+
+    // ARGUMENTS
+    NSArray* arguments = call.arguments[@"arguments"];
+    NSString* command = call.arguments[@"command"];
+    NSString* delimiter = call.arguments[@"delimiter"];
+
+    if ([@"getPlatform" isEqualToString:call.method]) {
+
+        NSString *architecture = [ArchDetect getArch];
+        result([FlutterFFmpegPlugin toStringDictionary:KEY_PLATFORM :[NSString stringWithFormat:@"%@-%@", PLATFORM_NAME, architecture]]);
+
+    } else if ([@"getFFmpegVersion" isEqualToString:call.method]) {
+
+        result([FlutterFFmpegPlugin toStringDictionary:KEY_VERSION :[MobileFFmpeg getFFmpegVersion]]);
+
+    } else if ([@"executeWithArguments" isEqualToString:call.method]) {
+
+        NSLog(@"Running FFmpeg with arguments: %@.\n", arguments);
+
+        dispatch_async(dispatch_get_global_queue(DISPATCH_QUEUE_PRIORITY_DEFAULT, 0), ^{
+
+            int rc = [MobileFFmpeg executeWithArguments:arguments];
+
+            NSLog(@"FFmpeg exited with rc: %d\n", rc);
+
+            result([FlutterFFmpegPlugin toIntDictionary:KEY_RC :[NSNumber numberWithInt:rc]]);
+        });
+
+    } else if ([@"execute" isEqualToString:call.method]) {
+
+        if (delimiter == nil) {
+            delimiter = @" ";
+        }
+
+        NSLog(@"Running FFmpeg command: %@ with delimiter %@.\n", command, delimiter);
+
+        dispatch_async(dispatch_get_global_queue(DISPATCH_QUEUE_PRIORITY_DEFAULT, 0), ^{
+            int rc = [MobileFFmpeg execute:command delimiter:delimiter];
+
+            NSLog(@"FFmpeg exited with rc: %d\n", rc);
+
+            result([FlutterFFmpegPlugin toIntDictionary:KEY_RC :[NSNumber numberWithInt:rc]]);
+        });
+
+    } else if ([@"cancel" isEqualToString:call.method]) {
+
+        [MobileFFmpeg cancel];
+
+    } else if ([@"enableRedirection" isEqualToString:call.method]) {
+
+        [MobileFFmpegConfig enableRedirection];
+
+    } else if ([@"disableRedirection" isEqualToString:call.method]) {
+
+        [MobileFFmpegConfig disableRedirection];
+
+    } else if ([@"getLogLevel" isEqualToString:call.method]) {
+
+        int logLevel = [MobileFFmpegConfig getLogLevel];
+        result([FlutterFFmpegPlugin toIntDictionary:KEY_LOG_LEVEL :[NSNumber numberWithInt:logLevel]]);
+
+    } else if ([@"setLogLevel" isEqualToString:call.method]) {
+
+        NSNumber* logLevel = call.arguments[@"level"];
+        [MobileFFmpegConfig setLogLevel:[logLevel intValue]];
+
+    } else if ([@"enableLogs" isEqualToString:call.method]) {
+
+        [MobileFFmpegConfig setLogDelegate:self];
+
+    } else if ([@"disableLogs" isEqualToString:call.method]) {
+
+        [MobileFFmpegConfig setLogDelegate:nil];
+
+    } else if ([@"enableStatistics" isEqualToString:call.method]) {
+
+        [MobileFFmpegConfig setStatisticsDelegate:self];
+
+    } else if ([@"disableStatistics" isEqualToString:call.method]) {
+
+        [MobileFFmpegConfig setStatisticsDelegate:nil];
+
+    } else if ([@"getLastReceivedStatistics" isEqualToString:call.method]) {
+
+        Statistics *statistics = [MobileFFmpegConfig getLastReceivedStatistics];
+        result([FlutterFFmpegPlugin toStatisticsDictionary:statistics]);
+
+    } else if ([@"resetStatistics" isEqualToString:call.method]) {
+
+        [MobileFFmpegConfig resetStatistics];
+
+    } else if ([@"setFontconfigConfigurationPath" isEqualToString:call.method]) {
+
+        NSString* path = call.arguments[@"path"];
+        [MobileFFmpegConfig setFontconfigConfigurationPath:path];
+
+    } else if ([@"setFontDirectory" isEqualToString:call.method]) {
+
+        NSString* fontDirectoryPath = call.arguments[@"fontDirectory"];
+        NSDictionary* fontNameMapping = call.arguments[@"fontNameMap"];
+        [MobileFFmpegConfig setFontDirectory:fontDirectoryPath with:fontNameMapping];
+
+    } else if ([@"getPackageName" isEqualToString:call.method]) {
+
+        NSString *packageName = [MobileFFmpegConfig getPackageName];
+        result([FlutterFFmpegPlugin toStringDictionary:KEY_PACKAGE_NAME :packageName]);
+
+    } else if ([@"getExternalLibraries" isEqualToString:call.method]) {
+
+        NSArray *externalLibraries = [MobileFFmpegConfig getExternalLibraries];
+        result(externalLibraries);
+
+    } else if ([@"getLastReturnCode" isEqualToString:call.method]) {
+
+        int lastReturnCode = [MobileFFmpeg getLastReturnCode];
+        result([FlutterFFmpegPlugin toIntDictionary:KEY_LAST_RC :[NSNumber numberWithInt:lastReturnCode]]);
+
+    } else if ([@"getLastCommandOutput" isEqualToString:call.method]) {
+
+        NSString *lastCommandOutput = [MobileFFmpeg getLastCommandOutput];
+        result([FlutterFFmpegPlugin toStringDictionary:KEY_LAST_COMMAND_OUTPUT :lastCommandOutput]);
+
+    } else if ([@"getMediaInformation" isEqualToString:call.method]) {
+
+        NSString* path = call.arguments[@"path"];
+        NSNumber* timeout = call.arguments[@"timeout"];
+
+        NSLog(@"Getting media information for %@ with timeout %d.\n", path, [timeout intValue]);
+
+        dispatch_async(dispatch_get_global_queue(DISPATCH_QUEUE_PRIORITY_DEFAULT, 0), ^{
+            MediaInformation *mediaInformation = [MobileFFmpeg getMediaInformation:path timeout:[timeout intValue]];
+            result([FlutterFFmpegPlugin toMediaInformationDictionary:mediaInformation]);
+        });
+
+    } else {
+
+        result(FlutterMethodNotImplemented);
+
+    }
+}
+
+- (void)logCallback: (int)level :(NSString*)message {
+    dispatch_async(dispatch_get_main_queue(), ^{
+        NSMutableDictionary *dictionary = [[NSMutableDictionary alloc] init];
+        dictionary[KEY_LOG_LEVEL] = [NSNumber numberWithInt:level];
+        dictionary[KEY_LOG_TEXT] = message;
+
+        [self emitLogMessage: dictionary];
+    });
+}
+
+- (void)statisticsCallback:(Statistics *)statistics {
+    dispatch_async(dispatch_get_main_queue(), ^{
+        [self emitStatistics: statistics];
+    });
+}
+
+- (void)emitLogMessage:(NSDictionary*)logMessage{
+    _eventSink([FlutterFFmpegPlugin toStringDictionary:EVENT_LOG :logMessage]);
+}
+
+- (void)emitStatistics:(Statistics*)statistics{
+    NSDictionary *dictionary = [FlutterFFmpegPlugin toStatisticsDictionary:statistics];
+    _eventSink([FlutterFFmpegPlugin toStringDictionary:EVENT_STAT :dictionary]);
+}
+
++ (NSDictionary *)toStringDictionary:(NSString*)key :(NSString*)value {
+    NSMutableDictionary *dictionary = [[NSMutableDictionary alloc] init];
+    dictionary[key] = value;
+
+    return dictionary;
+}
+
++ (NSDictionary *)toIntDictionary:(NSString*)key :(NSNumber*)value {
+    NSMutableDictionary *dictionary = [[NSMutableDictionary alloc] init];
+    dictionary[key] = value;
+
+    return dictionary;
+}
+
++ (NSDictionary *)toStatisticsDictionary:(Statistics*)statistics {
+    NSMutableDictionary *dictionary = [[NSMutableDictionary alloc] init];
+
+    if (statistics != nil) {
+        dictionary[KEY_STAT_TIME] = [NSNumber numberWithInt: [statistics getTime]];
+        dictionary[KEY_STAT_SIZE] = [NSNumber numberWithLong: [statistics getSize]];
+
+        dictionary[KEY_STAT_BITRATE] = [NSNumber numberWithDouble: [statistics getBitrate]];
+        dictionary[KEY_STAT_SPEED] = [NSNumber numberWithDouble: [statistics getSpeed]];
+
+        dictionary[KEY_STAT_VIDEO_FRAME_NUMBER] = [NSNumber numberWithInt: [statistics getVideoFrameNumber]];
+        dictionary[KEY_STAT_VIDEO_QUALITY] = [NSNumber numberWithFloat: [statistics getVideoQuality]];
+        dictionary[KEY_STAT_VIDEO_FPS] = [NSNumber numberWithFloat: [statistics getVideoFps]];
+    }
+
+    return dictionary;
+}
+
++ (NSDictionary *)toMediaInformationDictionary:(MediaInformation*)mediaInformation {
+    NSMutableDictionary *dictionary = [[NSMutableDictionary alloc] init];
+
+    if (mediaInformation != nil) {
+        if ([mediaInformation getFormat] != nil) {
+            dictionary[@"format"] =  [mediaInformation getFormat];
+        }
+        if ([mediaInformation getPath] != nil) {
+            dictionary[@"path"] = [mediaInformation getPath];
+        }
+        if ([mediaInformation getStartTime] != nil) {
+            dictionary[@"startTime"] = [mediaInformation getStartTime];
+        }
+        if ([mediaInformation getDuration] != nil) {
+            dictionary[@"duration"] = [mediaInformation getDuration];
+        }
+        if ([mediaInformation getBitrate] != nil) {
+            dictionary[@"bitrate"] = [mediaInformation getBitrate];
+        }
+        if ([mediaInformation getRawInformation] != nil) {
+            dictionary[@"rawInformation"] = [mediaInformation getRawInformation];
+        }
+
+        NSDictionary *metadataDictionary = [mediaInformation getMetadataEntries];
+        if (metadataDictionary != nil && ([metadataDictionary count] > 0)) {
+            dictionary[@"metadata"] = metadataDictionary;
+        }
+
+        NSArray *streams = [mediaInformation getStreams];
+        if (streams != nil && ([streams count] > 0)) {
+            NSMutableArray *array = [[NSMutableArray alloc] init];
+
+            for (int i=0; i < [streams count]; i++) {
+                StreamInformation *streamInformation= [streams objectAtIndex:i];
+                [array addObject: [FlutterFFmpegPlugin toStreamInformationDictionary:streamInformation]];
+            }
+
+            dictionary[@"streams"] = array;
+        }
+    }
+
+    return dictionary;
+}
+
++ (NSDictionary *)toStreamInformationDictionary:(StreamInformation*)streamInformation {
+    NSMutableDictionary *dictionary = [[NSMutableDictionary alloc] init];
+
+    if (streamInformation != nil) {
+        if ([streamInformation getIndex] != nil) {
+            dictionary[@"index"] = [streamInformation getIndex];
+        }
+        if ([streamInformation getType] != nil) {
+            dictionary[@"type"] = [streamInformation getType];
+        }
+        if ([streamInformation getCodec] != nil) {
+            dictionary[@"codec"] = [streamInformation getCodec];
+        }
+        if ([streamInformation getFullCodec] != nil) {
+            dictionary[@"fullCodec"] = [streamInformation getFullCodec];
+        }
+        if ([streamInformation getFormat] != nil) {
+            dictionary[@"format"] = [streamInformation getFormat];
+        }
+        if ([streamInformation getFullFormat] != nil) {
+            dictionary[@"fullFormat"] = [streamInformation getFullFormat];
+        }
+        if ([streamInformation getWidth] != nil) {
+            dictionary[@"width"] = [streamInformation getWidth];
+        }
+        if ([streamInformation getHeight] != nil) {
+            dictionary[@"height"] = [streamInformation getHeight];
+        }
+        if ([streamInformation getBitrate] != nil) {
+            dictionary[@"bitrate"] = [streamInformation getBitrate];
+        }
+        if ([streamInformation getSampleRate] != nil) {
+            dictionary[@"sampleRate"] = [streamInformation getSampleRate];
+        }
+        if ([streamInformation getSampleFormat] != nil) {
+            dictionary[@"sampleFormat"] = [streamInformation getSampleFormat];
+        }
+        if ([streamInformation getChannelLayout] != nil) {
+            dictionary[@"channelLayout"] = [streamInformation getChannelLayout];
+        }
+        if ([streamInformation getSampleAspectRatio] != nil) {
+            dictionary[@"sampleAspectRatio"] = [streamInformation getSampleAspectRatio];
+        }
+        if ([streamInformation getDisplayAspectRatio] != nil) {
+            dictionary[@"displayAspectRatio"] = [streamInformation getDisplayAspectRatio];
+        }
+        if ([streamInformation getAverageFrameRate] != nil) {
+            dictionary[@"averageFrameRate"] = [streamInformation getAverageFrameRate];
+        }
+        if ([streamInformation getRealFrameRate] != nil) {
+            dictionary[@"realFrameRate"] = [streamInformation getRealFrameRate];
+        }
+        if ([streamInformation getTimeBase] != nil) {
+            dictionary[@"timeBase"] = [streamInformation getTimeBase];
+        }
+        if ([streamInformation getCodecTimeBase] != nil) {
+            dictionary[@"codecTimeBase"] = [streamInformation getCodecTimeBase];
+        }
+
+        NSDictionary *metadataDictionary = [streamInformation getMetadataEntries];
+        if (metadataDictionary != nil && ([metadataDictionary count] > 0)) {
+            dictionary[@"metadata"] = metadataDictionary;
+        }
+    }
+
+    return dictionary;
+}
+
+@end

+ 22 - 0
packages/flutter_ffmpeg_full/ios/flutter_ffmpeg.podspec

@@ -0,0 +1,22 @@
+Pod::Spec.new do |s|
+  s.name             = 'flutter_ffmpeg'
+  s.version          = '0.1.0'
+  s.summary          = 'FFmpeg plugin for Flutter.'
+  s.description      = 'FFmpeg plugin based on mobile-ffmpeg for Flutter.'
+  s.homepage         = 'https://github.com/tanersener/flutter-ffmpeg'
+
+  s.author           = { 'Taner Sener' => 'tanersener@gmail.com' }
+  s.license          = { :file => '../LICENSE' }
+
+  s.requires_arc     = true
+  s.ios.deployment_target = '9.3'
+
+  s.source              = { :path => '.' }
+  s.source_files        = 'Classes/**/*'
+  s.public_header_files = 'Classes/**/*.h'
+
+  s.dependency      'Flutter'
+  s.dependency      'mobile-ffmpeg-full', '4.2.LTS'
+
+end
+

+ 352 - 0
packages/flutter_ffmpeg_full/lib/flutter_ffmpeg.dart

@@ -0,0 +1,352 @@
+/*
+ * Copyright (c) 2019 Taner Sener
+ *
+ * This file is part of FlutterFFmpeg.
+ *
+ * FlutterFFmpeg is free software: you can redistribute it and/or modify
+ * it under the terms of the GNU Lesser General Public License as published by
+ * the Free Software Foundation, either version 3 of the License, or
+ * (at your option) any later version.
+ *
+ * FlutterFFmpeg is distributed in the hope that it will be useful,
+ * but WITHOUT ANY WARRANTY; without even the implied warranty of
+ * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
+ * GNU Lesser General Public License for more details.
+ *
+ * You should have received a copy of the GNU Lesser General Public License
+ * along with FlutterFFmpeg.  If not, see <http://www.gnu.org/licenses/>.
+ */
+
+import 'dart:async';
+
+import 'package:flutter/services.dart';
+
+class FlutterFFmpeg {
+  static const MethodChannel _methodChannel = const MethodChannel('flutter_ffmpeg');
+  static const EventChannel _eventChannel = const EventChannel('flutter_ffmpeg_event');
+
+  Function(int level, String message) logCallback;
+  Function(int time, int size, double bitrate, double speed, int videoFrameNumber, double videoQuality, double videoFps) statisticsCallback;
+
+  FlutterFFmpeg() {
+    logCallback = null;
+    statisticsCallback = null;
+
+    print("Loading flutter-ffmpeg.");
+
+    _eventChannel.receiveBroadcastStream().listen(_onEvent, onError: _onError);
+
+    enableLogs();
+    enableStatistics();
+    enableRedirection();
+
+    getPlatform().then((name) => print("Loaded flutter-ffmpeg-$name."));
+  }
+
+  void _onEvent(Object event) {
+    if (event is Map<dynamic, dynamic>) {
+      final Map<String, dynamic> eventMap = event.cast();
+      final Map<dynamic, dynamic> logEvent = eventMap['FlutterFFmpegLogCallback'];
+      final Map<dynamic, dynamic> statisticsEvent = eventMap['FlutterFFmpegStatisticsCallback'];
+
+      if (logEvent != null) {
+        int level = logEvent['level'];
+        String message = logEvent['log'];
+
+        if (this.logCallback == null) {
+          if (message.length > 0) {
+            // PRINT ALREADY ADDS NEW LINE. SO REMOVE THIS ONE
+            if (message.endsWith('\n')) {
+              print(message.substring(0, message.length - 1));
+            } else {
+              print(message);
+            }
+          }
+        } else {
+          this.logCallback(level, message);
+        }
+      }
+
+      if (statisticsEvent != null) {
+        if (this.statisticsCallback != null) {
+          int time = statisticsEvent['time'];
+          int size = statisticsEvent['size'];
+          double bitrate = _doublePrecision(statisticsEvent['bitrate'], 2);
+          double speed = _doublePrecision(statisticsEvent['speed'], 2);
+          int videoFrameNumber = statisticsEvent['videoFrameNumber'];
+          double videoQuality = _doublePrecision(statisticsEvent['videoQuality'], 2);
+          double videoFps = _doublePrecision(statisticsEvent['videoFps'], 2);
+
+          this.statisticsCallback(time, size, bitrate, speed, videoFrameNumber, videoQuality, videoFps);
+        }
+      }
+    }
+  }
+
+  void _onError(Object error) {
+    print('Event error: $error');
+  }
+
+  double _doublePrecision(double value, int precision) {
+    if (value == null) {
+      return 0;
+    } else {
+      return num.parse(value.toStringAsFixed(precision));
+    }
+  }
+
+  /// Returns FFmpeg version bundled within the library.
+  Future<String> getFFmpegVersion() async {
+    try {
+      final Map<dynamic, dynamic> result = await _methodChannel.invokeMethod('getFFmpegVersion');
+      return result['version'];
+    } on PlatformException catch (e) {
+      print("Plugin error: ${e.message}");
+      return null;
+    }
+  }
+
+  /// Returns platform name where library is loaded.
+  Future<String> getPlatform() async {
+    try {
+      final Map<dynamic, dynamic> result = await _methodChannel.invokeMethod('getPlatform');
+      return result['platform'];
+    } on PlatformException catch (e) {
+      print("Plugin error: ${e.message}");
+      return null;
+    }
+  }
+
+  /// Executes FFmpeg with [commandArguments] provided.
+  Future<int> executeWithArguments(List<String> arguments) async {
+    try {
+      final Map<dynamic, dynamic> result = await _methodChannel.invokeMethod('executeWithArguments', {'arguments': arguments});
+      return result['rc'];
+    } on PlatformException catch (e) {
+      print("Plugin error: ${e.message}");
+      return -1;
+    }
+  }
+
+  /// Executes FFmpeg [command] provided. Command is split into arguments using provided [delimiter].
+  Future<int> execute(String command, [String delimiter = ' ']) async {
+    try {
+      final Map<dynamic, dynamic> result = await _methodChannel.invokeMethod('execute', {'command': command, 'delimiter': delimiter});
+      return result['rc'];
+    } on PlatformException catch (e) {
+      print("Plugin error: ${e.message}");
+      return -1;
+    }
+  }
+
+  /// Cancels an ongoing operation.
+  Future<void> cancel() async {
+    try {
+      await _methodChannel.invokeMethod('cancel');
+    } on PlatformException catch (e) {
+      print("Plugin error: ${e.message}");
+    }
+  }
+
+  /// Enables redirection
+  Future<void> enableRedirection() async {
+    try {
+      await _methodChannel.invokeMethod('enableRedirection');
+    } on PlatformException catch (e) {
+      print("Plugin error: ${e.message}");
+    }
+  }
+
+  /// Disables log and statistics redirection. By default redirection is enabled in constructor.
+  /// When redirection is enabled FFmpeg logs are printed to console and can be routed further to a callback function.
+  /// By disabling redirection, logs are redirected to stderr.
+  /// Statistics redirection behaviour is similar. Statistics are not printed at all if redirection is not enabled.
+  /// If it is enabled then it is possible to define a statistics callback function but if you don't, they are not
+  /// printed anywhere and only saved as codelastReceivedStatistics data which can be polled with
+  /// [getLastReceivedStatistics()].
+  Future<void> disableRedirection() async {
+    try {
+      await _methodChannel.invokeMethod('disableRedirection');
+    } on PlatformException catch (e) {
+      print("Plugin error: ${e.message}");
+    }
+  }
+
+  /// Returns log level.
+  Future<int> getLogLevel() async {
+    try {
+      final Map<dynamic, dynamic> result = await _methodChannel.invokeMethod('getLogLevel');
+      return result['level'];
+    } on PlatformException catch (e) {
+      print("Plugin error: ${e.message}");
+      return -1;
+    }
+  }
+
+  /// Sets log level.
+  Future<void> setLogLevel(int logLevel) async {
+    try {
+      await _methodChannel.invokeMethod('setLogLevel', {'level': logLevel});
+    } on PlatformException catch (e) {
+      print("Plugin error: ${e.message}");
+    }
+  }
+
+  /// Enables log events
+  Future<void> enableLogs() async {
+    try {
+      await _methodChannel.invokeMethod('enableLogs');
+    } on PlatformException catch (e) {
+      print("Plugin error: ${e.message}");
+    }
+  }
+
+  /// Disables log functionality of the library. Logs will not be printed to console and log callback will be disabled.
+  /// Note that log functionality is enabled by default.
+  Future<void> disableLogs() async {
+    try {
+      await _methodChannel.invokeMethod('disableLogs');
+    } on PlatformException catch (e) {
+      print("Plugin error: ${e.message}");
+    }
+  }
+
+  /// Enables statistics events.
+  Future<void> enableStatistics() async {
+    try {
+      await _methodChannel.invokeMethod('enableStatistics');
+    } on PlatformException catch (e) {
+      print("Plugin error: ${e.message}");
+    }
+  }
+
+  /// Disables statistics functionality of the library. Statistics callback will be disabled but the last received
+  /// statistics data will be still available.
+  /// Note that statistics functionality is enabled by default.
+  Future<void> disableStatistics() async {
+    try {
+      await _methodChannel.invokeMethod('disableStatistics');
+    } on PlatformException catch (e) {
+      print("Plugin error: ${e.message}");
+    }
+  }
+
+  /// Sets a callback to redirect FFmpeg logs. [newCallback] is a new log callback function, use null to disable a previously defined callback
+  void enableLogCallback(Function(int level, String message) newCallback) {
+    try {
+      this.logCallback = newCallback;
+    } on PlatformException catch (e) {
+      print("Plugin error: ${e.message}");
+    }
+  }
+
+  /// Sets a callback to redirect FFmpeg statistics. [newCallback] is a new statistics callback function, use null to disable a previously defined callback
+  void enableStatisticsCallback(Function(int time, int size, double bitrate, double speed, int videoFrameNumber, double videoQuality, double videoFps) newCallback) {
+    try {
+      this.statisticsCallback = newCallback;
+    } on PlatformException catch (e) {
+      print("Plugin error: ${e.message}");
+    }
+  }
+
+  /// Returns the last received statistics data stored in bitrate, size, speed, time, videoFps, videoFrameNumber and
+  /// videoQuality fields
+  Future<Map<dynamic, dynamic>> getLastReceivedStatistics() async {
+    try {
+      final Map<dynamic, dynamic> result = await _methodChannel.invokeMethod('getLastReceivedStatistics');
+      return result;
+    } on PlatformException catch (e) {
+      print("Plugin error: ${e.message}");
+      return null;
+    }
+  }
+
+  /// Resets last received statistics. It is recommended to call it before starting a new execution.
+  Future<void> resetStatistics() async {
+    try {
+      await _methodChannel.invokeMethod('resetStatistics');
+    } on PlatformException catch (e) {
+      print("Plugin error: ${e.message}");
+    }
+  }
+
+  /// Sets and overrides fontconfig configuration directory.
+  Future<void> setFontconfigConfigurationPath(String path) async {
+    try {
+      await _methodChannel.invokeMethod('setFontconfigConfigurationPath', {'path': path});
+    } on PlatformException catch (e) {
+      print("Plugin error: ${e.message}");
+    }
+  }
+
+  /// Registers fonts inside the given [fontDirectory], so they are available to use in FFmpeg filters.
+  Future<void> setFontDirectory(String fontDirectory, Map<String, String> fontNameMap) async {
+    var parameters;
+    if (fontNameMap == null) {
+      parameters = {'fontDirectory': fontDirectory};
+    } else {
+      parameters = {'fontDirectory': fontDirectory, 'fontNameMap': fontNameMap};
+    }
+
+    try {
+      await _methodChannel.invokeMethod('setFontDirectory', parameters);
+    } on PlatformException catch (e) {
+      print("Plugin error: ${e.message}");
+    }
+  }
+
+  /// Returns FlutterFFmpeg package name.
+  Future<String> getPackageName() async {
+    try {
+      final Map<dynamic, dynamic> result = await _methodChannel.invokeMethod('getPackageName');
+      return result['packageName'];
+    } on PlatformException catch (e) {
+      print("Plugin error: ${e.message}");
+      return null;
+    }
+  }
+
+  /// Returns supported external libraries.
+  Future<List<dynamic>> getExternalLibraries() async {
+    try {
+      final List<dynamic> result = await _methodChannel.invokeMethod('getExternalLibraries');
+      return result;
+    } on PlatformException catch (e) {
+      print("Plugin error: ${e.message}");
+      return null;
+    }
+  }
+
+  /// Returns return code of last executed command.
+  Future<int> getLastReturnCode() async {
+    try {
+      final Map<dynamic, dynamic> result = await _methodChannel.invokeMethod('getLastReturnCode');
+      return result['lastRc'];
+    } on PlatformException catch (e) {
+      print("Plugin error: ${e.message}");
+      return -1;
+    }
+  }
+
+  /// Returns log output of last executed command. Please note that disabling redirection using
+  /// [disableRedirection()] method also disables this functionality.
+  Future<String> getLastCommandOutput() async {
+    try {
+      final Map<dynamic, dynamic> result = await _methodChannel.invokeMethod('getLastCommandOutput');
+      return result['lastCommandOutput'];
+    } on PlatformException catch (e) {
+      print("Plugin error: ${e.message}");
+      return null;
+    }
+  }
+
+  /// Returns media information for given [path] using optional [timeout]
+  Future<Map<dynamic, dynamic>> getMediaInformation(String path, [int timeout = 10000]) async {
+    try {
+      return await _methodChannel.invokeMethod('getMediaInformation', {'path': path, 'timeout': timeout});
+    } on PlatformException catch (e) {
+      print("Plugin error: ${e.message}");
+      return null;
+    }
+  }
+}

+ 76 - 0
packages/flutter_ffmpeg_full/lib/log_level.dart

@@ -0,0 +1,76 @@
+/*
+ * Copyright (c) 2019 Taner Sener
+ *
+ * This file is part of FlutterFFmpeg.
+ *
+ * FlutterFFmpeg is free software: you can redistribute it and/or modify
+ * it under the terms of the GNU Lesser General Public License as published by
+ * the Free Software Foundation, either version 3 of the License, or
+ * (at your option) any later version.
+ *
+ * FlutterFFmpeg is distributed in the hope that it will be useful,
+ * but WITHOUT ANY WARRANTY; without even the implied warranty of
+ * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
+ * GNU Lesser General Public License for more details.
+ *
+ * You should have received a copy of the GNU Lesser General Public License
+ * along with FlutterFFmpeg.  If not, see <http://www.gnu.org/licenses/>.
+ */
+
+class LogLevel {
+  /// Print no output.
+  static const int AV_LOG_QUIET = -8;
+
+  /// Something went really wrong and we will crash now.
+  static const int AV_LOG_PANIC = 0;
+
+  /// Something went wrong and recovery is not possible.
+  /// For example, no header was found for a format which depends
+  /// on headers or an illegal combination of parameters is used.
+  static const int AV_LOG_FATAL = 8;
+
+  /// Something went wrong and cannot losslessly be recovered.
+  /// However, not all future data is affected.
+  static const int AV_LOG_ERROR = 16;
+
+  /// Something somehow does not look correct. This may or may not
+  /// lead to problems. An example would be the use of '-vstrict -2'.
+  static const int AV_LOG_WARNING = 24;
+
+  /// int Standard information.
+  static const int AV_LOG_INFO = 32;
+
+  /// Detailed information.
+  static const int AV_LOG_VERBOSE = 40;
+
+  /// Stuff which is only useful for libav* developers.
+  static const int AV_LOG_DEBUG = 48;
+
+  /// Extremely verbose debugging, useful for libav* development.
+  static const int AV_LOG_TRACE = 56;
+
+  /// Returns log level string from int
+  static String levelToString(int level) {
+    switch (level) {
+      case LogLevel.AV_LOG_TRACE:
+        return "TRACE";
+      case LogLevel.AV_LOG_DEBUG:
+        return "DEBUG";
+      case LogLevel.AV_LOG_VERBOSE:
+        return "VERBOSE";
+      case LogLevel.AV_LOG_INFO:
+        return "INFO";
+      case LogLevel.AV_LOG_WARNING:
+        return "WARNING";
+      case LogLevel.AV_LOG_ERROR:
+        return "ERROR";
+      case LogLevel.AV_LOG_FATAL:
+        return "FATAL";
+      case LogLevel.AV_LOG_PANIC:
+        return "PANIC";
+      case LogLevel.AV_LOG_QUIET:
+      default:
+        return "";
+    }
+  }
+}

+ 17 - 0
packages/flutter_ffmpeg_full/pubspec.yaml

@@ -0,0 +1,17 @@
+name: flutter_ffmpeg
+description: FFmpeg plugin for Flutter.
+version: 0.1.0
+author: Taner Sener <tanersener@gmail.com>
+homepage: https://github.com/tanersener/flutter-ffmpeg
+
+environment:
+  sdk: ">=2.0.0-dev.68.0 <3.0.0"
+
+dependencies:
+  flutter:
+    sdk: flutter
+
+flutter:
+  plugin:
+    androidPackage: com.arthenica.flutter.ffmpeg
+    pluginClass: FlutterFFmpegPlugin

+ 7 - 0
packages/flutter_ffmpeg_https-gpl/.gitignore

@@ -0,0 +1,7 @@
+.idea/
+.DS_Store
+.dart_tool/
+.pub/
+build/
+pubspec.lock
+.packages

+ 10 - 0
packages/flutter_ffmpeg_https-gpl/.metadata

@@ -0,0 +1,10 @@
+# This file tracks properties of this Flutter project.
+# Used by Flutter tool to assess capabilities and perform upgrades etc.
+#
+# This file should be version controlled and should not be manually edited.
+
+version:
+  revision: 5391447fae6209bb21a89e6a5a6583cac1af9b4b
+  channel: stable
+
+project_type: plugin

+ 674 - 0
packages/flutter_ffmpeg_https-gpl/LICENSE

@@ -0,0 +1,674 @@
+                    GNU GENERAL PUBLIC LICENSE
+                       Version 3, 29 June 2007
+
+ Copyright (C) 2007 Free Software Foundation, Inc. <http://fsf.org/>
+ Everyone is permitted to copy and distribute verbatim copies
+ of this license document, but changing it is not allowed.
+
+                            Preamble
+
+  The GNU General Public License is a free, copyleft license for
+software and other kinds of works.
+
+  The licenses for most software and other practical works are designed
+to take away your freedom to share and change the works.  By contrast,
+the GNU General Public License is intended to guarantee your freedom to
+share and change all versions of a program--to make sure it remains free
+software for all its users.  We, the Free Software Foundation, use the
+GNU General Public License for most of our software; it applies also to
+any other work released this way by its authors.  You can apply it to
+your programs, too.
+
+  When we speak of free software, we are referring to freedom, not
+price.  Our General Public Licenses are designed to make sure that you
+have the freedom to distribute copies of free software (and charge for
+them if you wish), that you receive source code or can get it if you
+want it, that you can change the software or use pieces of it in new
+free programs, and that you know you can do these things.
+
+  To protect your rights, we need to prevent others from denying you
+these rights or asking you to surrender the rights.  Therefore, you have
+certain responsibilities if you distribute copies of the software, or if
+you modify it: responsibilities to respect the freedom of others.
+
+  For example, if you distribute copies of such a program, whether
+gratis or for a fee, you must pass on to the recipients the same
+freedoms that you received.  You must make sure that they, too, receive
+or can get the source code.  And you must show them these terms so they
+know their rights.
+
+  Developers that use the GNU GPL protect your rights with two steps:
+(1) assert copyright on the software, and (2) offer you this License
+giving you legal permission to copy, distribute and/or modify it.
+
+  For the developers' and authors' protection, the GPL clearly explains
+that there is no warranty for this free software.  For both users' and
+authors' sake, the GPL requires that modified versions be marked as
+changed, so that their problems will not be attributed erroneously to
+authors of previous versions.
+
+  Some devices are designed to deny users access to install or run
+modified versions of the software inside them, although the manufacturer
+can do so.  This is fundamentally incompatible with the aim of
+protecting users' freedom to change the software.  The systematic
+pattern of such abuse occurs in the area of products for individuals to
+use, which is precisely where it is most unacceptable.  Therefore, we
+have designed this version of the GPL to prohibit the practice for those
+products.  If such problems arise substantially in other domains, we
+stand ready to extend this provision to those domains in future versions
+of the GPL, as needed to protect the freedom of users.
+
+  Finally, every program is threatened constantly by software patents.
+States should not allow patents to restrict development and use of
+software on general-purpose computers, but in those that do, we wish to
+avoid the special danger that patents applied to a free program could
+make it effectively proprietary.  To prevent this, the GPL assures that
+patents cannot be used to render the program non-free.
+
+  The precise terms and conditions for copying, distribution and
+modification follow.
+
+                       TERMS AND CONDITIONS
+
+  0. Definitions.
+
+  "This License" refers to version 3 of the GNU General Public License.
+
+  "Copyright" also means copyright-like laws that apply to other kinds of
+works, such as semiconductor masks.
+
+  "The Program" refers to any copyrightable work licensed under this
+License.  Each licensee is addressed as "you".  "Licensees" and
+"recipients" may be individuals or organizations.
+
+  To "modify" a work means to copy from or adapt all or part of the work
+in a fashion requiring copyright permission, other than the making of an
+exact copy.  The resulting work is called a "modified version" of the
+earlier work or a work "based on" the earlier work.
+
+  A "covered work" means either the unmodified Program or a work based
+on the Program.
+
+  To "propagate" a work means to do anything with it that, without
+permission, would make you directly or secondarily liable for
+infringement under applicable copyright law, except executing it on a
+computer or modifying a private copy.  Propagation includes copying,
+distribution (with or without modification), making available to the
+public, and in some countries other activities as well.
+
+  To "convey" a work means any kind of propagation that enables other
+parties to make or receive copies.  Mere interaction with a user through
+a computer network, with no transfer of a copy, is not conveying.
+
+  An interactive user interface displays "Appropriate Legal Notices"
+to the extent that it includes a convenient and prominently visible
+feature that (1) displays an appropriate copyright notice, and (2)
+tells the user that there is no warranty for the work (except to the
+extent that warranties are provided), that licensees may convey the
+work under this License, and how to view a copy of this License.  If
+the interface presents a list of user commands or options, such as a
+menu, a prominent item in the list meets this criterion.
+
+  1. Source Code.
+
+  The "source code" for a work means the preferred form of the work
+for making modifications to it.  "Object code" means any non-source
+form of a work.
+
+  A "Standard Interface" means an interface that either is an official
+standard defined by a recognized standards body, or, in the case of
+interfaces specified for a particular programming language, one that
+is widely used among developers working in that language.
+
+  The "System Libraries" of an executable work include anything, other
+than the work as a whole, that (a) is included in the normal form of
+packaging a Major Component, but which is not part of that Major
+Component, and (b) serves only to enable use of the work with that
+Major Component, or to implement a Standard Interface for which an
+implementation is available to the public in source code form.  A
+"Major Component", in this context, means a major essential component
+(kernel, window system, and so on) of the specific operating system
+(if any) on which the executable work runs, or a compiler used to
+produce the work, or an object code interpreter used to run it.
+
+  The "Corresponding Source" for a work in object code form means all
+the source code needed to generate, install, and (for an executable
+work) run the object code and to modify the work, including scripts to
+control those activities.  However, it does not include the work's
+System Libraries, or general-purpose tools or generally available free
+programs which are used unmodified in performing those activities but
+which are not part of the work.  For example, Corresponding Source
+includes interface definition files associated with source files for
+the work, and the source code for shared libraries and dynamically
+linked subprograms that the work is specifically designed to require,
+such as by intimate data communication or control flow between those
+subprograms and other parts of the work.
+
+  The Corresponding Source need not include anything that users
+can regenerate automatically from other parts of the Corresponding
+Source.
+
+  The Corresponding Source for a work in source code form is that
+same work.
+
+  2. Basic Permissions.
+
+  All rights granted under this License are granted for the term of
+copyright on the Program, and are irrevocable provided the stated
+conditions are met.  This License explicitly affirms your unlimited
+permission to run the unmodified Program.  The output from running a
+covered work is covered by this License only if the output, given its
+content, constitutes a covered work.  This License acknowledges your
+rights of fair use or other equivalent, as provided by copyright law.
+
+  You may make, run and propagate covered works that you do not
+convey, without conditions so long as your license otherwise remains
+in force.  You may convey covered works to others for the sole purpose
+of having them make modifications exclusively for you, or provide you
+with facilities for running those works, provided that you comply with
+the terms of this License in conveying all material for which you do
+not control copyright.  Those thus making or running the covered works
+for you must do so exclusively on your behalf, under your direction
+and control, on terms that prohibit them from making any copies of
+your copyrighted material outside their relationship with you.
+
+  Conveying under any other circumstances is permitted solely under
+the conditions stated below.  Sublicensing is not allowed; section 10
+makes it unnecessary.
+
+  3. Protecting Users' Legal Rights From Anti-Circumvention Law.
+
+  No covered work shall be deemed part of an effective technological
+measure under any applicable law fulfilling obligations under article
+11 of the WIPO copyright treaty adopted on 20 December 1996, or
+similar laws prohibiting or restricting circumvention of such
+measures.
+
+  When you convey a covered work, you waive any legal power to forbid
+circumvention of technological measures to the extent such circumvention
+is effected by exercising rights under this License with respect to
+the covered work, and you disclaim any intention to limit operation or
+modification of the work as a means of enforcing, against the work's
+users, your or third parties' legal rights to forbid circumvention of
+technological measures.
+
+  4. Conveying Verbatim Copies.
+
+  You may convey verbatim copies of the Program's source code as you
+receive it, in any medium, provided that you conspicuously and
+appropriately publish on each copy an appropriate copyright notice;
+keep intact all notices stating that this License and any
+non-permissive terms added in accord with section 7 apply to the code;
+keep intact all notices of the absence of any warranty; and give all
+recipients a copy of this License along with the Program.
+
+  You may charge any price or no price for each copy that you convey,
+and you may offer support or warranty protection for a fee.
+
+  5. Conveying Modified Source Versions.
+
+  You may convey a work based on the Program, or the modifications to
+produce it from the Program, in the form of source code under the
+terms of section 4, provided that you also meet all of these conditions:
+
+    a) The work must carry prominent notices stating that you modified
+    it, and giving a relevant date.
+
+    b) The work must carry prominent notices stating that it is
+    released under this License and any conditions added under section
+    7.  This requirement modifies the requirement in section 4 to
+    "keep intact all notices".
+
+    c) You must license the entire work, as a whole, under this
+    License to anyone who comes into possession of a copy.  This
+    License will therefore apply, along with any applicable section 7
+    additional terms, to the whole of the work, and all its parts,
+    regardless of how they are packaged.  This License gives no
+    permission to license the work in any other way, but it does not
+    invalidate such permission if you have separately received it.
+
+    d) If the work has interactive user interfaces, each must display
+    Appropriate Legal Notices; however, if the Program has interactive
+    interfaces that do not display Appropriate Legal Notices, your
+    work need not make them do so.
+
+  A compilation of a covered work with other separate and independent
+works, which are not by their nature extensions of the covered work,
+and which are not combined with it such as to form a larger program,
+in or on a volume of a storage or distribution medium, is called an
+"aggregate" if the compilation and its resulting copyright are not
+used to limit the access or legal rights of the compilation's users
+beyond what the individual works permit.  Inclusion of a covered work
+in an aggregate does not cause this License to apply to the other
+parts of the aggregate.
+
+  6. Conveying Non-Source Forms.
+
+  You may convey a covered work in object code form under the terms
+of sections 4 and 5, provided that you also convey the
+machine-readable Corresponding Source under the terms of this License,
+in one of these ways:
+
+    a) Convey the object code in, or embodied in, a physical product
+    (including a physical distribution medium), accompanied by the
+    Corresponding Source fixed on a durable physical medium
+    customarily used for software interchange.
+
+    b) Convey the object code in, or embodied in, a physical product
+    (including a physical distribution medium), accompanied by a
+    written offer, valid for at least three years and valid for as
+    long as you offer spare parts or customer support for that product
+    model, to give anyone who possesses the object code either (1) a
+    copy of the Corresponding Source for all the software in the
+    product that is covered by this License, on a durable physical
+    medium customarily used for software interchange, for a price no
+    more than your reasonable cost of physically performing this
+    conveying of source, or (2) access to copy the
+    Corresponding Source from a network server at no charge.
+
+    c) Convey individual copies of the object code with a copy of the
+    written offer to provide the Corresponding Source.  This
+    alternative is allowed only occasionally and noncommercially, and
+    only if you received the object code with such an offer, in accord
+    with subsection 6b.
+
+    d) Convey the object code by offering access from a designated
+    place (gratis or for a charge), and offer equivalent access to the
+    Corresponding Source in the same way through the same place at no
+    further charge.  You need not require recipients to copy the
+    Corresponding Source along with the object code.  If the place to
+    copy the object code is a network server, the Corresponding Source
+    may be on a different server (operated by you or a third party)
+    that supports equivalent copying facilities, provided you maintain
+    clear directions next to the object code saying where to find the
+    Corresponding Source.  Regardless of what server hosts the
+    Corresponding Source, you remain obligated to ensure that it is
+    available for as long as needed to satisfy these requirements.
+
+    e) Convey the object code using peer-to-peer transmission, provided
+    you inform other peers where the object code and Corresponding
+    Source of the work are being offered to the general public at no
+    charge under subsection 6d.
+
+  A separable portion of the object code, whose source code is excluded
+from the Corresponding Source as a System Library, need not be
+included in conveying the object code work.
+
+  A "User Product" is either (1) a "consumer product", which means any
+tangible personal property which is normally used for personal, family,
+or household purposes, or (2) anything designed or sold for incorporation
+into a dwelling.  In determining whether a product is a consumer product,
+doubtful cases shall be resolved in favor of coverage.  For a particular
+product received by a particular user, "normally used" refers to a
+typical or common use of that class of product, regardless of the status
+of the particular user or of the way in which the particular user
+actually uses, or expects or is expected to use, the product.  A product
+is a consumer product regardless of whether the product has substantial
+commercial, industrial or non-consumer uses, unless such uses represent
+the only significant mode of use of the product.
+
+  "Installation Information" for a User Product means any methods,
+procedures, authorization keys, or other information required to install
+and execute modified versions of a covered work in that User Product from
+a modified version of its Corresponding Source.  The information must
+suffice to ensure that the continued functioning of the modified object
+code is in no case prevented or interfered with solely because
+modification has been made.
+
+  If you convey an object code work under this section in, or with, or
+specifically for use in, a User Product, and the conveying occurs as
+part of a transaction in which the right of possession and use of the
+User Product is transferred to the recipient in perpetuity or for a
+fixed term (regardless of how the transaction is characterized), the
+Corresponding Source conveyed under this section must be accompanied
+by the Installation Information.  But this requirement does not apply
+if neither you nor any third party retains the ability to install
+modified object code on the User Product (for example, the work has
+been installed in ROM).
+
+  The requirement to provide Installation Information does not include a
+requirement to continue to provide support service, warranty, or updates
+for a work that has been modified or installed by the recipient, or for
+the User Product in which it has been modified or installed.  Access to a
+network may be denied when the modification itself materially and
+adversely affects the operation of the network or violates the rules and
+protocols for communication across the network.
+
+  Corresponding Source conveyed, and Installation Information provided,
+in accord with this section must be in a format that is publicly
+documented (and with an implementation available to the public in
+source code form), and must require no special password or key for
+unpacking, reading or copying.
+
+  7. Additional Terms.
+
+  "Additional permissions" are terms that supplement the terms of this
+License by making exceptions from one or more of its conditions.
+Additional permissions that are applicable to the entire Program shall
+be treated as though they were included in this License, to the extent
+that they are valid under applicable law.  If additional permissions
+apply only to part of the Program, that part may be used separately
+under those permissions, but the entire Program remains governed by
+this License without regard to the additional permissions.
+
+  When you convey a copy of a covered work, you may at your option
+remove any additional permissions from that copy, or from any part of
+it.  (Additional permissions may be written to require their own
+removal in certain cases when you modify the work.)  You may place
+additional permissions on material, added by you to a covered work,
+for which you have or can give appropriate copyright permission.
+
+  Notwithstanding any other provision of this License, for material you
+add to a covered work, you may (if authorized by the copyright holders of
+that material) supplement the terms of this License with terms:
+
+    a) Disclaiming warranty or limiting liability differently from the
+    terms of sections 15 and 16 of this License; or
+
+    b) Requiring preservation of specified reasonable legal notices or
+    author attributions in that material or in the Appropriate Legal
+    Notices displayed by works containing it; or
+
+    c) Prohibiting misrepresentation of the origin of that material, or
+    requiring that modified versions of such material be marked in
+    reasonable ways as different from the original version; or
+
+    d) Limiting the use for publicity purposes of names of licensors or
+    authors of the material; or
+
+    e) Declining to grant rights under trademark law for use of some
+    trade names, trademarks, or service marks; or
+
+    f) Requiring indemnification of licensors and authors of that
+    material by anyone who conveys the material (or modified versions of
+    it) with contractual assumptions of liability to the recipient, for
+    any liability that these contractual assumptions directly impose on
+    those licensors and authors.
+
+  All other non-permissive additional terms are considered "further
+restrictions" within the meaning of section 10.  If the Program as you
+received it, or any part of it, contains a notice stating that it is
+governed by this License along with a term that is a further
+restriction, you may remove that term.  If a license document contains
+a further restriction but permits relicensing or conveying under this
+License, you may add to a covered work material governed by the terms
+of that license document, provided that the further restriction does
+not survive such relicensing or conveying.
+
+  If you add terms to a covered work in accord with this section, you
+must place, in the relevant source files, a statement of the
+additional terms that apply to those files, or a notice indicating
+where to find the applicable terms.
+
+  Additional terms, permissive or non-permissive, may be stated in the
+form of a separately written license, or stated as exceptions;
+the above requirements apply either way.
+
+  8. Termination.
+
+  You may not propagate or modify a covered work except as expressly
+provided under this License.  Any attempt otherwise to propagate or
+modify it is void, and will automatically terminate your rights under
+this License (including any patent licenses granted under the third
+paragraph of section 11).
+
+  However, if you cease all violation of this License, then your
+license from a particular copyright holder is reinstated (a)
+provisionally, unless and until the copyright holder explicitly and
+finally terminates your license, and (b) permanently, if the copyright
+holder fails to notify you of the violation by some reasonable means
+prior to 60 days after the cessation.
+
+  Moreover, your license from a particular copyright holder is
+reinstated permanently if the copyright holder notifies you of the
+violation by some reasonable means, this is the first time you have
+received notice of violation of this License (for any work) from that
+copyright holder, and you cure the violation prior to 30 days after
+your receipt of the notice.
+
+  Termination of your rights under this section does not terminate the
+licenses of parties who have received copies or rights from you under
+this License.  If your rights have been terminated and not permanently
+reinstated, you do not qualify to receive new licenses for the same
+material under section 10.
+
+  9. Acceptance Not Required for Having Copies.
+
+  You are not required to accept this License in order to receive or
+run a copy of the Program.  Ancillary propagation of a covered work
+occurring solely as a consequence of using peer-to-peer transmission
+to receive a copy likewise does not require acceptance.  However,
+nothing other than this License grants you permission to propagate or
+modify any covered work.  These actions infringe copyright if you do
+not accept this License.  Therefore, by modifying or propagating a
+covered work, you indicate your acceptance of this License to do so.
+
+  10. Automatic Licensing of Downstream Recipients.
+
+  Each time you convey a covered work, the recipient automatically
+receives a license from the original licensors, to run, modify and
+propagate that work, subject to this License.  You are not responsible
+for enforcing compliance by third parties with this License.
+
+  An "entity transaction" is a transaction transferring control of an
+organization, or substantially all assets of one, or subdividing an
+organization, or merging organizations.  If propagation of a covered
+work results from an entity transaction, each party to that
+transaction who receives a copy of the work also receives whatever
+licenses to the work the party's predecessor in interest had or could
+give under the previous paragraph, plus a right to possession of the
+Corresponding Source of the work from the predecessor in interest, if
+the predecessor has it or can get it with reasonable efforts.
+
+  You may not impose any further restrictions on the exercise of the
+rights granted or affirmed under this License.  For example, you may
+not impose a license fee, royalty, or other charge for exercise of
+rights granted under this License, and you may not initiate litigation
+(including a cross-claim or counterclaim in a lawsuit) alleging that
+any patent claim is infringed by making, using, selling, offering for
+sale, or importing the Program or any portion of it.
+
+  11. Patents.
+
+  A "contributor" is a copyright holder who authorizes use under this
+License of the Program or a work on which the Program is based.  The
+work thus licensed is called the contributor's "contributor version".
+
+  A contributor's "essential patent claims" are all patent claims
+owned or controlled by the contributor, whether already acquired or
+hereafter acquired, that would be infringed by some manner, permitted
+by this License, of making, using, or selling its contributor version,
+but do not include claims that would be infringed only as a
+consequence of further modification of the contributor version.  For
+purposes of this definition, "control" includes the right to grant
+patent sublicenses in a manner consistent with the requirements of
+this License.
+
+  Each contributor grants you a non-exclusive, worldwide, royalty-free
+patent license under the contributor's essential patent claims, to
+make, use, sell, offer for sale, import and otherwise run, modify and
+propagate the contents of its contributor version.
+
+  In the following three paragraphs, a "patent license" is any express
+agreement or commitment, however denominated, not to enforce a patent
+(such as an express permission to practice a patent or covenant not to
+sue for patent infringement).  To "grant" such a patent license to a
+party means to make such an agreement or commitment not to enforce a
+patent against the party.
+
+  If you convey a covered work, knowingly relying on a patent license,
+and the Corresponding Source of the work is not available for anyone
+to copy, free of charge and under the terms of this License, through a
+publicly available network server or other readily accessible means,
+then you must either (1) cause the Corresponding Source to be so
+available, or (2) arrange to deprive yourself of the benefit of the
+patent license for this particular work, or (3) arrange, in a manner
+consistent with the requirements of this License, to extend the patent
+license to downstream recipients.  "Knowingly relying" means you have
+actual knowledge that, but for the patent license, your conveying the
+covered work in a country, or your recipient's use of the covered work
+in a country, would infringe one or more identifiable patents in that
+country that you have reason to believe are valid.
+
+  If, pursuant to or in connection with a single transaction or
+arrangement, you convey, or propagate by procuring conveyance of, a
+covered work, and grant a patent license to some of the parties
+receiving the covered work authorizing them to use, propagate, modify
+or convey a specific copy of the covered work, then the patent license
+you grant is automatically extended to all recipients of the covered
+work and works based on it.
+
+  A patent license is "discriminatory" if it does not include within
+the scope of its coverage, prohibits the exercise of, or is
+conditioned on the non-exercise of one or more of the rights that are
+specifically granted under this License.  You may not convey a covered
+work if you are a party to an arrangement with a third party that is
+in the business of distributing software, under which you make payment
+to the third party based on the extent of your activity of conveying
+the work, and under which the third party grants, to any of the
+parties who would receive the covered work from you, a discriminatory
+patent license (a) in connection with copies of the covered work
+conveyed by you (or copies made from those copies), or (b) primarily
+for and in connection with specific products or compilations that
+contain the covered work, unless you entered into that arrangement,
+or that patent license was granted, prior to 28 March 2007.
+
+  Nothing in this License shall be construed as excluding or limiting
+any implied license or other defenses to infringement that may
+otherwise be available to you under applicable patent law.
+
+  12. No Surrender of Others' Freedom.
+
+  If conditions are imposed on you (whether by court order, agreement or
+otherwise) that contradict the conditions of this License, they do not
+excuse you from the conditions of this License.  If you cannot convey a
+covered work so as to satisfy simultaneously your obligations under this
+License and any other pertinent obligations, then as a consequence you may
+not convey it at all.  For example, if you agree to terms that obligate you
+to collect a royalty for further conveying from those to whom you convey
+the Program, the only way you could satisfy both those terms and this
+License would be to refrain entirely from conveying the Program.
+
+  13. Use with the GNU Affero General Public License.
+
+  Notwithstanding any other provision of this License, you have
+permission to link or combine any covered work with a work licensed
+under version 3 of the GNU Affero General Public License into a single
+combined work, and to convey the resulting work.  The terms of this
+License will continue to apply to the part which is the covered work,
+but the special requirements of the GNU Affero General Public License,
+section 13, concerning interaction through a network will apply to the
+combination as such.
+
+  14. Revised Versions of this License.
+
+  The Free Software Foundation may publish revised and/or new versions of
+the GNU General Public License from time to time.  Such new versions will
+be similar in spirit to the present version, but may differ in detail to
+address new problems or concerns.
+
+  Each version is given a distinguishing version number.  If the
+Program specifies that a certain numbered version of the GNU General
+Public License "or any later version" applies to it, you have the
+option of following the terms and conditions either of that numbered
+version or of any later version published by the Free Software
+Foundation.  If the Program does not specify a version number of the
+GNU General Public License, you may choose any version ever published
+by the Free Software Foundation.
+
+  If the Program specifies that a proxy can decide which future
+versions of the GNU General Public License can be used, that proxy's
+public statement of acceptance of a version permanently authorizes you
+to choose that version for the Program.
+
+  Later license versions may give you additional or different
+permissions.  However, no additional obligations are imposed on any
+author or copyright holder as a result of your choosing to follow a
+later version.
+
+  15. Disclaimer of Warranty.
+
+  THERE IS NO WARRANTY FOR THE PROGRAM, TO THE EXTENT PERMITTED BY
+APPLICABLE LAW.  EXCEPT WHEN OTHERWISE STATED IN WRITING THE COPYRIGHT
+HOLDERS AND/OR OTHER PARTIES PROVIDE THE PROGRAM "AS IS" WITHOUT WARRANTY
+OF ANY KIND, EITHER EXPRESSED OR IMPLIED, INCLUDING, BUT NOT LIMITED TO,
+THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR
+PURPOSE.  THE ENTIRE RISK AS TO THE QUALITY AND PERFORMANCE OF THE PROGRAM
+IS WITH YOU.  SHOULD THE PROGRAM PROVE DEFECTIVE, YOU ASSUME THE COST OF
+ALL NECESSARY SERVICING, REPAIR OR CORRECTION.
+
+  16. Limitation of Liability.
+
+  IN NO EVENT UNLESS REQUIRED BY APPLICABLE LAW OR AGREED TO IN WRITING
+WILL ANY COPYRIGHT HOLDER, OR ANY OTHER PARTY WHO MODIFIES AND/OR CONVEYS
+THE PROGRAM AS PERMITTED ABOVE, BE LIABLE TO YOU FOR DAMAGES, INCLUDING ANY
+GENERAL, SPECIAL, INCIDENTAL OR CONSEQUENTIAL DAMAGES ARISING OUT OF THE
+USE OR INABILITY TO USE THE PROGRAM (INCLUDING BUT NOT LIMITED TO LOSS OF
+DATA OR DATA BEING RENDERED INACCURATE OR LOSSES SUSTAINED BY YOU OR THIRD
+PARTIES OR A FAILURE OF THE PROGRAM TO OPERATE WITH ANY OTHER PROGRAMS),
+EVEN IF SUCH HOLDER OR OTHER PARTY HAS BEEN ADVISED OF THE POSSIBILITY OF
+SUCH DAMAGES.
+
+  17. Interpretation of Sections 15 and 16.
+
+  If the disclaimer of warranty and limitation of liability provided
+above cannot be given local legal effect according to their terms,
+reviewing courts shall apply local law that most closely approximates
+an absolute waiver of all civil liability in connection with the
+Program, unless a warranty or assumption of liability accompanies a
+copy of the Program in return for a fee.
+
+                     END OF TERMS AND CONDITIONS
+
+            How to Apply These Terms to Your New Programs
+
+  If you develop a new program, and you want it to be of the greatest
+possible use to the public, the best way to achieve this is to make it
+free software which everyone can redistribute and change under these terms.
+
+  To do so, attach the following notices to the program.  It is safest
+to attach them to the start of each source file to most effectively
+state the exclusion of warranty; and each file should have at least
+the "copyright" line and a pointer to where the full notice is found.
+
+    <one line to give the program's name and a brief idea of what it does.>
+    Copyright (C) <year>  <name of author>
+
+    This program is free software: you can redistribute it and/or modify
+    it under the terms of the GNU General Public License as published by
+    the Free Software Foundation, either version 3 of the License, or
+    (at your option) any later version.
+
+    This program is distributed in the hope that it will be useful,
+    but WITHOUT ANY WARRANTY; without even the implied warranty of
+    MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
+    GNU General Public License for more details.
+
+    You should have received a copy of the GNU General Public License
+    along with this program.  If not, see <http://www.gnu.org/licenses/>.
+
+Also add information on how to contact you by electronic and paper mail.
+
+  If the program does terminal interaction, make it output a short
+notice like this when it starts in an interactive mode:
+
+    <program>  Copyright (C) <year>  <name of author>
+    This program comes with ABSOLUTELY NO WARRANTY; for details type `show w'.
+    This is free software, and you are welcome to redistribute it
+    under certain conditions; type `show c' for details.
+
+The hypothetical commands `show w' and `show c' should show the appropriate
+parts of the General Public License.  Of course, your program's commands
+might be different; for a GUI interface, you would use an "about box".
+
+  You should also get your employer (if you work as a programmer) or school,
+if any, to sign a "copyright disclaimer" for the program, if necessary.
+For more information on this, and how to apply and follow the GNU GPL, see
+<http://www.gnu.org/licenses/>.
+
+  The GNU General Public License does not permit incorporating your program
+into proprietary programs.  If your program is a subroutine library, you
+may consider it more useful to permit linking proprietary applications with
+the library.  If this is what you want to do, use the GNU Lesser General
+Public License instead of this License.  But first, please read
+<http://www.gnu.org/philosophy/why-not-lgpl.html>.

+ 8 - 0
packages/flutter_ffmpeg_https-gpl/android/.gitignore

@@ -0,0 +1,8 @@
+*.iml
+.gradle
+/local.properties
+/.idea/workspace.xml
+/.idea/libraries
+.DS_Store
+/build
+/captures

+ 38 - 0
packages/flutter_ffmpeg_https-gpl/android/build.gradle

@@ -0,0 +1,38 @@
+group 'com.arthenica.flutter.ffmpeg'
+version '0.1.0'
+
+buildscript {
+    repositories {
+        google()
+        jcenter()
+    }
+
+    dependencies {
+        classpath 'com.android.tools.build:gradle:3.2.1'
+    }
+}
+
+rootProject.allprojects {
+    repositories {
+        google()
+        jcenter()
+    }
+}
+
+apply plugin: 'com.android.library'
+
+android {
+    compileSdkVersion 27
+
+    defaultConfig {
+        minSdkVersion 21
+        testInstrumentationRunner "android.support.test.runner.AndroidJUnitRunner"
+    }
+    lintOptions {
+        disable 'InvalidPackage'
+    }
+}
+
+dependencies {
+    implementation 'com.arthenica:mobile-ffmpeg-https-gpl:4.2.LTS'
+}

+ 1 - 0
packages/flutter_ffmpeg_https-gpl/android/gradle.properties

@@ -0,0 +1 @@
+org.gradle.jvmargs=-Xmx1536M

+ 1 - 0
packages/flutter_ffmpeg_https-gpl/android/settings.gradle

@@ -0,0 +1 @@
+rootProject.name = 'flutter_ffmpeg'

+ 3 - 0
packages/flutter_ffmpeg_https-gpl/android/src/main/AndroidManifest.xml

@@ -0,0 +1,3 @@
+<manifest xmlns:android="http://schemas.android.com/apk/res/android"
+  package="com.arthenica.flutter.ffmpeg">
+</manifest>

+ 66 - 0
packages/flutter_ffmpeg_https-gpl/android/src/main/java/com/arthenica/flutter/ffmpeg/FlutterFFmpegExecuteAsyncArgumentsTask.java

@@ -0,0 +1,66 @@
+/*
+ * Copyright (c) 2019 Taner Sener
+ *
+ * This file is part of FlutterFFmpeg.
+ *
+ * FlutterFFmpeg is free software: you can redistribute it and/or modify
+ * it under the terms of the GNU Lesser General Public License as published by
+ * the Free Software Foundation, either version 3 of the License, or
+ * (at your option) any later version.
+ *
+ * FlutterFFmpeg is distributed in the hope that it will be useful,
+ * but WITHOUT ANY WARRANTY; without even the implied warranty of
+ * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
+ * GNU Lesser General Public License for more details.
+ *
+ * You should have received a copy of the GNU Lesser General Public License
+ * along with FlutterFFmpeg.  If not, see <http://www.gnu.org/licenses/>.
+ */
+
+package com.arthenica.flutter.ffmpeg;
+
+import android.os.AsyncTask;
+import android.util.Log;
+
+import com.arthenica.mobileffmpeg.FFmpeg;
+
+import java.util.Arrays;
+import java.util.List;
+
+import io.flutter.plugin.common.MethodChannel;
+
+/**
+ * Asynchronous task which performs {@link FFmpeg#execute(String[])} method invocations.
+ *
+ * @author Taner Sener
+ * @since 0.1.0
+ */
+public class FlutterFFmpegExecuteAsyncArgumentsTask extends AsyncTask<String, Integer, Integer> {
+
+    private final MethodChannel.Result result;
+    private final List<String> arguments;
+
+    FlutterFFmpegExecuteAsyncArgumentsTask(final List<String> arguments, final MethodChannel.Result result) {
+        this.arguments = arguments;
+        this.result = result;
+    }
+
+    @Override
+    protected Integer doInBackground(final String... dummyString) {
+        final String[] argumentsArray = arguments.toArray(new String[0]);
+
+        Log.d(FlutterFFmpegPlugin.LIBRARY_NAME, String.format("Running FFmpeg with arguments: %s.", Arrays.toString(argumentsArray)));
+
+        int rc = FFmpeg.execute(argumentsArray);
+
+        Log.d(FlutterFFmpegPlugin.LIBRARY_NAME, String.format("FFmpeg exited with rc: %d", rc));
+
+        return rc;
+    }
+
+    @Override
+    protected void onPostExecute(final Integer rc) {
+        result.success(FlutterFFmpegPlugin.toIntMap(FlutterFFmpegPlugin.KEY_RC, rc));
+    }
+
+}

+ 72 - 0
packages/flutter_ffmpeg_https-gpl/android/src/main/java/com/arthenica/flutter/ffmpeg/FlutterFFmpegExecuteAsyncCommandTask.java

@@ -0,0 +1,72 @@
+/*
+ * Copyright (c) 2019 Taner Sener
+ *
+ * This file is part of FlutterFFmpeg.
+ *
+ * FlutterFFmpeg is free software: you can redistribute it and/or modify
+ * it under the terms of the GNU Lesser General Public License as published by
+ * the Free Software Foundation, either version 3 of the License, or
+ * (at your option) any later version.
+ *
+ * FlutterFFmpeg is distributed in the hope that it will be useful,
+ * but WITHOUT ANY WARRANTY; without even the implied warranty of
+ * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
+ * GNU Lesser General Public License for more details.
+ *
+ * You should have received a copy of the GNU Lesser General Public License
+ * along with FlutterFFmpeg.  If not, see <http://www.gnu.org/licenses/>.
+ */
+
+package com.arthenica.flutter.ffmpeg;
+
+import android.os.AsyncTask;
+import android.util.Log;
+
+import com.arthenica.mobileffmpeg.FFmpeg;
+
+import io.flutter.plugin.common.MethodChannel;
+
+/**
+ * Asynchronous task which performs {@link FFmpeg#execute(String, String)} method invocations.
+ *
+ * @author Taner Sener
+ * @since 0.1.0
+ */
+public class FlutterFFmpegExecuteAsyncCommandTask extends AsyncTask<String, Integer, Integer> {
+
+    private String delimiter;
+    private final MethodChannel.Result result;
+
+    FlutterFFmpegExecuteAsyncCommandTask(final String delimiter, final MethodChannel.Result result) {
+        if (delimiter == null) {
+            this.delimiter = " ";
+        } else {
+            this.delimiter = delimiter;
+        }
+
+        this.result = result;
+    }
+
+    @Override
+    protected Integer doInBackground(final String... strings) {
+        int rc = -1;
+
+        if ((strings != null) && (strings.length > 0)) {
+            final String command = strings[0];
+
+            Log.d(FlutterFFmpegPlugin.LIBRARY_NAME, String.format("Running FFmpeg command: %s with delimiter %s.", command, delimiter));
+
+            rc = FFmpeg.execute(command, delimiter);
+
+            Log.d(FlutterFFmpegPlugin.LIBRARY_NAME, String.format("FFmpeg exited with rc: %d", rc));
+        }
+
+        return rc;
+    }
+
+    @Override
+    protected void onPostExecute(final Integer rc) {
+        result.success(FlutterFFmpegPlugin.toIntMap(FlutterFFmpegPlugin.KEY_RC, rc));
+    }
+
+}

+ 70 - 0
packages/flutter_ffmpeg_https-gpl/android/src/main/java/com/arthenica/flutter/ffmpeg/FlutterFFmpegGetMediaInformationAsyncTask.java

@@ -0,0 +1,70 @@
+/*
+ * Copyright (c) 2019 Taner Sener
+ *
+ * This file is part of FlutterFFmpeg.
+ *
+ * FlutterFFmpeg is free software: you can redistribute it and/or modify
+ * it under the terms of the GNU Lesser General Public License as published by
+ * the Free Software Foundation, either version 3 of the License, or
+ * (at your option) any later version.
+ *
+ * FlutterFFmpeg is distributed in the hope that it will be useful,
+ * but WITHOUT ANY WARRANTY; without even the implied warranty of
+ * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
+ * GNU Lesser General Public License for more details.
+ *
+ * You should have received a copy of the GNU Lesser General Public License
+ * along with FlutterFFmpeg.  If not, see <http://www.gnu.org/licenses/>.
+ */
+
+package com.arthenica.flutter.ffmpeg;
+
+import android.os.AsyncTask;
+import android.util.Log;
+
+import com.arthenica.mobileffmpeg.FFmpeg;
+import com.arthenica.mobileffmpeg.MediaInformation;
+
+import io.flutter.plugin.common.MethodChannel;
+
+/**
+ * Asynchronous task which performs {@link FFmpeg#getMediaInformation(String, Long)} method invocations.
+ *
+ * @author Taner Sener
+ * @since 0.1.0
+ */
+public class FlutterFFmpegGetMediaInformationAsyncTask extends AsyncTask<String, Integer, MediaInformation> {
+
+    private Integer timeout;
+    private final MethodChannel.Result result;
+
+    FlutterFFmpegGetMediaInformationAsyncTask(final Integer timeout, final MethodChannel.Result result) {
+        this.timeout = timeout;
+        this.result = result;
+    }
+
+    @Override
+    protected MediaInformation doInBackground(final String... strings) {
+        MediaInformation mediaInformation = null;
+
+        if ((strings != null) && (strings.length > 0)) {
+            final String path = strings[0];
+
+            if (timeout == null) {
+                Log.d(FlutterFFmpegPlugin.LIBRARY_NAME, String.format("Getting media information for %s", path));
+                mediaInformation = FFmpeg.getMediaInformation(path);
+            } else {
+                Log.d(FlutterFFmpegPlugin.LIBRARY_NAME, String.format("Getting media information for %s with timeout %d.", path, timeout.longValue()));
+                mediaInformation = FFmpeg.getMediaInformation(path, timeout.longValue());
+            }
+        }
+
+        return mediaInformation;
+    }
+
+    @Override
+    protected void onPostExecute(final MediaInformation mediaInformation) {
+        result.success(FlutterFFmpegPlugin.toMediaInformationMap(mediaInformation));
+    }
+
+}

+ 436 - 0
packages/flutter_ffmpeg_https-gpl/android/src/main/java/com/arthenica/flutter/ffmpeg/FlutterFFmpegPlugin.java

@@ -0,0 +1,436 @@
+/*
+ * Copyright (c) 2019 Taner Sener
+ *
+ * This file is part of FlutterFFmpeg.
+ *
+ * FlutterFFmpeg is free software: you can redistribute it and/or modify
+ * it under the terms of the GNU Lesser General Public License as published by
+ * the Free Software Foundation, either version 3 of the License, or
+ * (at your option) any later version.
+ *
+ * FlutterFFmpeg is distributed in the hope that it will be useful,
+ * but WITHOUT ANY WARRANTY; without even the implied warranty of
+ * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
+ * GNU Lesser General Public License for more details.
+ *
+ * You should have received a copy of the GNU Lesser General Public License
+ * along with FlutterFFmpeg.  If not, see <http://www.gnu.org/licenses/>.
+ */
+
+package com.arthenica.flutter.ffmpeg;
+
+import android.content.Context;
+import android.system.ErrnoException;
+import android.util.Log;
+
+import com.arthenica.mobileffmpeg.AbiDetect;
+import com.arthenica.mobileffmpeg.Config;
+import com.arthenica.mobileffmpeg.FFmpeg;
+import com.arthenica.mobileffmpeg.Level;
+import com.arthenica.mobileffmpeg.LogCallback;
+import com.arthenica.mobileffmpeg.LogMessage;
+import com.arthenica.mobileffmpeg.MediaInformation;
+import com.arthenica.mobileffmpeg.Statistics;
+import com.arthenica.mobileffmpeg.StatisticsCallback;
+import com.arthenica.mobileffmpeg.StreamInformation;
+
+import java.util.ArrayList;
+import java.util.HashMap;
+import java.util.List;
+import java.util.Map;
+import java.util.Set;
+
+import io.flutter.plugin.common.EventChannel;
+import io.flutter.plugin.common.MethodCall;
+import io.flutter.plugin.common.MethodChannel;
+import io.flutter.plugin.common.MethodChannel.MethodCallHandler;
+import io.flutter.plugin.common.MethodChannel.Result;
+import io.flutter.plugin.common.PluginRegistry.Registrar;
+
+/**
+ * <h3>Flutter FFmpeg Plugin</h3>
+ *
+ * @author Taner Sener
+ * @since 0.1.0
+ */
+public class FlutterFFmpegPlugin implements MethodCallHandler, EventChannel.StreamHandler {
+    public static final String LIBRARY_NAME = "flutter-ffmpeg";
+
+    public static final String PLATFORM_NAME = "android";
+    public static final String KEY_VERSION = "version";
+    public static final String KEY_RC = "rc";
+    public static final String KEY_PLATFORM = "platform";
+    public static final String KEY_PACKAGE_NAME = "packageName";
+    public static final String KEY_LAST_RC = "lastRc";
+
+    public static final String KEY_LAST_COMMAND_OUTPUT = "lastCommandOutput";
+    public static final String KEY_LOG_TEXT = "log";
+
+    public static final String KEY_LOG_LEVEL = "level";
+    public static final String KEY_STAT_TIME = "time";
+    public static final String KEY_STAT_SIZE = "size";
+    public static final String KEY_STAT_BITRATE = "bitrate";
+    public static final String KEY_STAT_SPEED = "speed";
+    public static final String KEY_STAT_VIDEO_FRAME_NUMBER = "videoFrameNumber";
+    public static final String KEY_STAT_VIDEO_QUALITY = "videoQuality";
+    public static final String KEY_STAT_VIDEO_FPS = "videoFps";
+
+    public static final String EVENT_LOG = "FlutterFFmpegLogCallback";
+    public static final String EVENT_STAT = "FlutterFFmpegStatisticsCallback";
+
+    private EventChannel.EventSink eventSink;
+    private final Registrar registrar;
+
+    /**
+     * Registers plugin to registry.
+     *
+     * @param registrar receiver of plugin registration
+     */
+    public static void registerWith(final Registrar registrar) {
+        FlutterFFmpegPlugin handler = new FlutterFFmpegPlugin(registrar);
+
+        final MethodChannel channel = new MethodChannel(registrar.messenger(), "flutter_ffmpeg");
+        channel.setMethodCallHandler(handler);
+
+        final EventChannel eventChannel = new EventChannel(registrar.messenger(), "flutter_ffmpeg_event");
+        eventChannel.setStreamHandler(handler);
+    }
+
+    private FlutterFFmpegPlugin(Registrar registrar) {
+        this.registrar = registrar;
+    }
+
+    private Context getActiveContext() {
+        return (registrar.activity() != null) ? registrar.activity() : registrar.context();
+    }
+
+    /**
+     * Handles method calls.
+     *
+     * @param call   method call
+     * @param result result callback
+     */
+    @Override
+    public void onMethodCall(final MethodCall call, final Result result) {
+        if (call.method.equals("getPlatform")) {
+
+            final String abi = AbiDetect.getAbi();
+            result.success(toStringMap(KEY_PLATFORM, PLATFORM_NAME + "-" + abi));
+
+        } else if (call.method.equals("getFFmpegVersion")) {
+
+            final String version = FFmpeg.getFFmpegVersion();
+            result.success(toStringMap(KEY_VERSION, version));
+
+        } else if (call.method.equals("executeWithArguments")) {
+
+            List<String> arguments = call.argument("arguments");
+
+            final FlutterFFmpegExecuteAsyncArgumentsTask asyncTask = new FlutterFFmpegExecuteAsyncArgumentsTask(arguments, result);
+            asyncTask.execute("dummy-trigger");
+
+        } else if (call.method.equals("execute")) {
+
+            String command = call.argument("command");
+            String delimiter = call.argument("delimiter");
+
+            final FlutterFFmpegExecuteAsyncCommandTask asyncTask = new FlutterFFmpegExecuteAsyncCommandTask(delimiter, result);
+            asyncTask.execute(command);
+
+        } else if (call.method.equals("cancel")) {
+
+            FFmpeg.cancel();
+
+        } else if (call.method.equals("enableRedirection")) {
+
+            Config.enableRedirection();
+
+        } else if (call.method.equals("disableRedirection")) {
+
+            Config.disableRedirection();
+
+        } else if (call.method.equals("getLogLevel")) {
+
+            final Level level = Config.getLogLevel();
+            result.success(toIntMap(KEY_LOG_LEVEL, levelToInt(level)));
+
+        } else if (call.method.equals("setLogLevel")) {
+
+            Integer level = call.argument("level");
+            if (level == null) {
+                level = Level.AV_LOG_TRACE.getValue();
+            }
+            Config.setLogLevel(Level.from(level));
+
+        } else if (call.method.equals("enableLogs")) {
+
+            Config.enableLogCallback(new LogCallback() {
+
+                @Override
+                public void apply(final LogMessage logMessage) {
+                    emitLogMessage(logMessage);
+                }
+            });
+
+        } else if (call.method.equals("disableLogs")) {
+
+            Config.enableLogCallback(null);
+
+        } else if (call.method.equals("enableStatistics")) {
+
+            Config.enableStatisticsCallback(new StatisticsCallback() {
+
+                @Override
+                public void apply(final Statistics statistics) {
+                    emitStatistics(statistics);
+                }
+            });
+
+        } else if (call.method.equals("disableStatistics")) {
+
+            Config.enableStatisticsCallback(null);
+
+        } else if (call.method.equals("getLastReceivedStatistics")) {
+
+            result.success(toMap(Config.getLastReceivedStatistics()));
+
+        } else if (call.method.equals("resetStatistics")) {
+
+            Config.resetStatistics();
+
+        } else if (call.method.equals("setFontconfigConfigurationPath")) {
+            String path = call.argument("path");
+
+            try {
+                Config.setFontconfigConfigurationPath(path);
+            } catch (final ErrnoException e) {
+                Log.w(LIBRARY_NAME, String.format("Setting fontconfig configuration path failed for %s", path), e);
+            }
+
+        } else if (call.method.equals("setFontDirectory")) {
+
+            String path = call.argument("fontDirectory");
+            Map<String, String> map = call.argument("fontNameMap");
+
+            Config.setFontDirectory(getActiveContext(), path, map);
+
+        } else if (call.method.equals("getPackageName")) {
+
+            final String packageName = Config.getPackageName();
+            result.success(toStringMap(KEY_PACKAGE_NAME, packageName));
+
+        } else if (call.method.equals("getExternalLibraries")) {
+
+            final List<String> externalLibraries = Config.getExternalLibraries();
+            result.success(externalLibraries);
+
+        } else if (call.method.equals("getLastReturnCode")) {
+
+            int lastReturnCode = FFmpeg.getLastReturnCode();
+            result.success(toIntMap(KEY_LAST_RC, lastReturnCode));
+
+        } else if (call.method.equals("getLastCommandOutput")) {
+
+            final String lastCommandOutput = FFmpeg.getLastCommandOutput();
+            result.success(toStringMap(KEY_LAST_COMMAND_OUTPUT, lastCommandOutput));
+
+        } else if (call.method.equals("getMediaInformation")) {
+            final String path = call.argument("path");
+            Integer timeout = call.argument("timeout");
+            if (timeout == null) {
+                timeout = 10000;
+            }
+
+            final FlutterFFmpegGetMediaInformationAsyncTask asyncTask = new FlutterFFmpegGetMediaInformationAsyncTask(timeout, result);
+            asyncTask.execute(path);
+
+        } else {
+            result.notImplemented();
+        }
+    }
+
+    @Override
+    public void onListen(Object o, EventChannel.EventSink eventSink) {
+        this.eventSink = eventSink;
+    }
+
+    @Override
+    public void onCancel(Object o) {
+        this.eventSink = null;
+    }
+
+    protected void emitLogMessage(final LogMessage logMessage) {
+        final HashMap<String, Object> logWrapperMap = new HashMap<>();
+        final HashMap<String, Object> logMap = new HashMap<>();
+
+        logMap.put(KEY_LOG_LEVEL, levelToInt(logMessage.getLevel()));
+        logMap.put(KEY_LOG_TEXT, logMessage.getText());
+
+        logWrapperMap.put(EVENT_LOG, logMap);
+
+        eventSink.success(logWrapperMap);
+    }
+
+    protected void emitStatistics(final Statistics statistics) {
+        final HashMap<String, Object> statisticsMap = new HashMap<>();
+        statisticsMap.put(EVENT_STAT, toMap(statistics));
+        eventSink.success(statisticsMap);
+    }
+
+    public static int levelToInt(final Level level) {
+        return (level == null) ? Level.AV_LOG_TRACE.getValue() : level.getValue();
+    }
+
+    public static HashMap<String, String> toStringMap(final String key, final String value) {
+        final HashMap<String, String> map = new HashMap<>();
+        map.put(key, value);
+        return map;
+    }
+
+    public static HashMap<String, Integer> toIntMap(final String key, final int value) {
+        final HashMap<String, Integer> map = new HashMap<>();
+        map.put(key, value);
+        return map;
+    }
+
+    public static Map<String, Object> toMap(final Statistics statistics) {
+        final HashMap<String, Object> statisticsMap = new HashMap<>();
+
+        if (statistics != null) {
+            statisticsMap.put(KEY_STAT_TIME, statistics.getTime());
+            statisticsMap.put(KEY_STAT_SIZE, (statistics.getSize() < Integer.MAX_VALUE) ? (int) statistics.getSize() : (int) (statistics.getSize() % Integer.MAX_VALUE));
+            statisticsMap.put(KEY_STAT_BITRATE, statistics.getBitrate());
+            statisticsMap.put(KEY_STAT_SPEED, statistics.getSpeed());
+
+            statisticsMap.put(KEY_STAT_VIDEO_FRAME_NUMBER, statistics.getVideoFrameNumber());
+            statisticsMap.put(KEY_STAT_VIDEO_QUALITY, statistics.getVideoQuality());
+            statisticsMap.put(KEY_STAT_VIDEO_FPS, statistics.getVideoFps());
+        }
+
+        return statisticsMap;
+    }
+
+    public static HashMap<String, Object> toMediaInformationMap(final MediaInformation mediaInformation) {
+        final HashMap<String, Object> map = new HashMap<>();
+
+        if (mediaInformation != null) {
+            if (mediaInformation.getFormat() != null) {
+                map.put("format", mediaInformation.getFormat());
+            }
+            if (mediaInformation.getPath() != null) {
+                map.put("path", mediaInformation.getPath());
+            }
+            if (mediaInformation.getStartTime() != null) {
+                map.put("startTime", mediaInformation.getStartTime().intValue());
+            }
+            if (mediaInformation.getDuration() != null) {
+                map.put("duration", mediaInformation.getDuration().intValue());
+            }
+            if (mediaInformation.getBitrate() != null) {
+                map.put("bitrate", mediaInformation.getBitrate().intValue());
+            }
+            if (mediaInformation.getRawInformation() != null) {
+                map.put("rawInformation", mediaInformation.getRawInformation());
+            }
+
+            final Set<Map.Entry<String, String>> metadata = mediaInformation.getMetadataEntries();
+            if ((metadata != null) && (metadata.size() > 0)) {
+                final HashMap<String, String> metadataMap = new HashMap<>();
+
+                for (Map.Entry<String, String> entry : metadata) {
+                    metadataMap.put(entry.getKey(), entry.getValue());
+                }
+
+                map.put("metadata", metadataMap);
+            }
+
+            final List<StreamInformation> streams = mediaInformation.getStreams();
+            if ((streams != null) && (streams.size() > 0)) {
+                final ArrayList<Map<String, Object>> array = new ArrayList<>();
+
+                for (StreamInformation streamInformation : streams) {
+                    array.add(toStreamInformationMap(streamInformation));
+                }
+
+                map.put("streams", array);
+            }
+        }
+
+        return map;
+    }
+
+    public static Map<String, Object> toStreamInformationMap(final StreamInformation streamInformation) {
+        final HashMap<String, Object> map = new HashMap<>();
+
+        if (streamInformation != null) {
+            if (streamInformation.getIndex() != null) {
+                map.put("index", streamInformation.getIndex().intValue());
+            }
+            if (streamInformation.getType() != null) {
+                map.put("type", streamInformation.getType());
+            }
+            if (streamInformation.getCodec() != null) {
+                map.put("codec", streamInformation.getCodec());
+            }
+            if (streamInformation.getFullCodec() != null) {
+                map.put("fullCodec", streamInformation.getFullCodec());
+            }
+            if (streamInformation.getFormat() != null) {
+                map.put("format", streamInformation.getFormat());
+            }
+            if (streamInformation.getFullFormat() != null) {
+                map.put("fullFormat", streamInformation.getFullFormat());
+            }
+            if (streamInformation.getWidth() != null) {
+                map.put("width", streamInformation.getWidth().intValue());
+            }
+            if (streamInformation.getHeight() != null) {
+                map.put("height", streamInformation.getHeight().intValue());
+            }
+            if (streamInformation.getBitrate() != null) {
+                map.put("bitrate", streamInformation.getBitrate().intValue());
+            }
+            if (streamInformation.getSampleRate() != null) {
+                map.put("sampleRate", streamInformation.getSampleRate().intValue());
+            }
+            if (streamInformation.getSampleFormat() != null) {
+                map.put("sampleFormat", streamInformation.getSampleFormat());
+            }
+            if (streamInformation.getChannelLayout() != null) {
+                map.put("channelLayout", streamInformation.getChannelLayout());
+            }
+            if (streamInformation.getSampleAspectRatio() != null) {
+                map.put("sampleAspectRatio", streamInformation.getSampleAspectRatio());
+            }
+            if (streamInformation.getDisplayAspectRatio() != null) {
+                map.put("displayAspectRatio", streamInformation.getDisplayAspectRatio());
+            }
+            if (streamInformation.getAverageFrameRate() != null) {
+                map.put("averageFrameRate", streamInformation.getAverageFrameRate());
+            }
+            if (streamInformation.getRealFrameRate() != null) {
+                map.put("realFrameRate", streamInformation.getRealFrameRate());
+            }
+            if (streamInformation.getTimeBase() != null) {
+                map.put("timeBase", streamInformation.getTimeBase());
+            }
+            if (streamInformation.getCodecTimeBase() != null) {
+                map.put("codecTimeBase", streamInformation.getCodecTimeBase());
+            }
+
+            final Set<Map.Entry<String, String>> metadata = streamInformation.getMetadataEntries();
+            if ((metadata != null) && (metadata.size() > 0)) {
+                final HashMap<String, String> metadataMap = new HashMap<>();
+
+                for (Map.Entry<String, String> entry : metadata) {
+                    metadataMap.put(entry.getKey(), entry.getValue());
+                }
+
+                map.put("metadata", metadataMap);
+            }
+        }
+
+        return map;
+    }
+
+}

+ 36 - 0
packages/flutter_ffmpeg_https-gpl/ios/.gitignore

@@ -0,0 +1,36 @@
+.idea/
+.vagrant/
+.sconsign.dblite
+.svn/
+
+.DS_Store
+*.swp
+profile
+
+DerivedData/
+build/
+GeneratedPluginRegistrant.h
+GeneratedPluginRegistrant.m
+
+.generated/
+
+*.pbxuser
+*.mode1v3
+*.mode2v3
+*.perspectivev3
+
+!default.pbxuser
+!default.mode1v3
+!default.mode2v3
+!default.perspectivev3
+
+xcuserdata
+
+*.moved-aside
+
+*.pyc
+*sync/
+Icon?
+.tags*
+
+/Flutter/Generated.xcconfig

+ 0 - 0
packages/flutter_ffmpeg_https-gpl/ios/Assets/.gitkeep


+ 27 - 0
packages/flutter_ffmpeg_https-gpl/ios/Classes/FlutterFFmpegPlugin.h

@@ -0,0 +1,27 @@
+/*
+ * Copyright (c) 2019 Taner Sener
+ *
+ * This file is part of FlutterFFmpeg.
+ *
+ * FlutterFFmpeg is free software: you can redistribute it and/or modify
+ * it under the terms of the GNU Lesser General Public License as published by
+ * the Free Software Foundation, either version 3 of the License, or
+ * (at your option) any later version.
+ *
+ * FlutterFFmpeg is distributed in the hope that it will be useful,
+ * but WITHOUT ANY WARRANTY; without even the implied warranty of
+ * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
+ * GNU Lesser General Public License for more details.
+ *
+ * You should have received a copy of the GNU Lesser General Public License
+ * along with FlutterFFmpeg.  If not, see <http://www.gnu.org/licenses/>.
+ */
+
+#import <Flutter/Flutter.h>
+#import <mobileffmpeg/MobileFFmpegConfig.h>
+
+/**
+ * Flutter FFmpeg Plugin
+ */
+@interface FlutterFFmpegPlugin : NSObject<FlutterPlugin,FlutterStreamHandler,LogDelegate,StatisticsDelegate>
+@end

+ 387 - 0
packages/flutter_ffmpeg_https-gpl/ios/Classes/FlutterFfmpegPlugin.m

@@ -0,0 +1,387 @@
+/*
+ * Copyright (c) 2019 Taner Sener
+ *
+ * This file is part of FlutterFFmpeg.
+ *
+ * FlutterFFmpeg is free software: you can redistribute it and/or modify
+ * it under the terms of the GNU Lesser General Public License as published by
+ * the Free Software Foundation, either version 3 of the License, or
+ * (at your option) any later version.
+ *
+ * FlutterFFmpeg is distributed in the hope that it will be useful,
+ * but WITHOUT ANY WARRANTY; without even the implied warranty of
+ * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
+ * GNU Lesser General Public License for more details.
+ *
+ * You should have received a copy of the GNU Lesser General Public License
+ * along with FlutterFFmpeg.  If not, see <http://www.gnu.org/licenses/>.
+ */
+
+#import "FlutterFFmpegPlugin.h"
+
+#import <mobileffmpeg/ArchDetect.h>
+#import <mobileffmpeg/MobileFFmpeg.h>
+#import <mobileffmpeg/MobileFFmpegConfig.h>
+
+static NSString *const PLATFORM_NAME = @"ios";
+
+static NSString *const KEY_VERSION = @"version";
+static NSString *const KEY_RC = @"rc";
+static NSString *const KEY_PLATFORM = @"platform";
+static NSString *const KEY_PACKAGE_NAME = @"packageName";
+static NSString *const KEY_LAST_RC = @"lastRc";
+static NSString *const KEY_LAST_COMMAND_OUTPUT = @"lastCommandOutput";
+
+static NSString *const KEY_LOG_TEXT = @"log";
+static NSString *const KEY_LOG_LEVEL = @"level";
+
+static NSString *const KEY_STAT_TIME = @"time";
+static NSString *const KEY_STAT_SIZE = @"size";
+static NSString *const KEY_STAT_BITRATE = @"bitrate";
+static NSString *const KEY_STAT_SPEED = @"speed";
+static NSString *const KEY_STAT_VIDEO_FRAME_NUMBER = @"videoFrameNumber";
+static NSString *const KEY_STAT_VIDEO_QUALITY = @"videoQuality";
+static NSString *const KEY_STAT_VIDEO_FPS = @"videoFps";
+
+static NSString *const EVENT_LOG = @"FlutterFFmpegLogCallback";
+static NSString *const EVENT_STAT = @"FlutterFFmpegStatisticsCallback";
+
+/**
+ * Flutter FFmpeg Plugin
+ */
+@implementation FlutterFFmpegPlugin {
+    FlutterEventSink _eventSink;
+}
+
+- (FlutterError *)onListenWithArguments:(id)arguments eventSink:(FlutterEventSink)eventSink {
+    _eventSink = eventSink;
+    return nil;
+}
+
+- (FlutterError *)onCancelWithArguments:(id)arguments {
+    _eventSink = nil;
+    return nil;
+}
+
++ (void)registerWithRegistrar:(NSObject<FlutterPluginRegistrar>*)registrar {
+    FlutterFFmpegPlugin* instance = [[FlutterFFmpegPlugin alloc] init];
+
+    FlutterMethodChannel* methodChannel = [FlutterMethodChannel methodChannelWithName:@"flutter_ffmpeg" binaryMessenger:[registrar messenger]];
+    [registrar addMethodCallDelegate:instance channel:methodChannel];
+
+    FlutterEventChannel* eventChannel = [FlutterEventChannel eventChannelWithName:@"flutter_ffmpeg_event" binaryMessenger:[registrar messenger]];
+    [eventChannel setStreamHandler:instance];
+}
+
+- (void)handleMethodCall:(FlutterMethodCall*)call result:(FlutterResult)result {
+
+    // ARGUMENTS
+    NSArray* arguments = call.arguments[@"arguments"];
+    NSString* command = call.arguments[@"command"];
+    NSString* delimiter = call.arguments[@"delimiter"];
+
+    if ([@"getPlatform" isEqualToString:call.method]) {
+
+        NSString *architecture = [ArchDetect getArch];
+        result([FlutterFFmpegPlugin toStringDictionary:KEY_PLATFORM :[NSString stringWithFormat:@"%@-%@", PLATFORM_NAME, architecture]]);
+
+    } else if ([@"getFFmpegVersion" isEqualToString:call.method]) {
+
+        result([FlutterFFmpegPlugin toStringDictionary:KEY_VERSION :[MobileFFmpeg getFFmpegVersion]]);
+
+    } else if ([@"executeWithArguments" isEqualToString:call.method]) {
+
+        NSLog(@"Running FFmpeg with arguments: %@.\n", arguments);
+
+        dispatch_async(dispatch_get_global_queue(DISPATCH_QUEUE_PRIORITY_DEFAULT, 0), ^{
+
+            int rc = [MobileFFmpeg executeWithArguments:arguments];
+
+            NSLog(@"FFmpeg exited with rc: %d\n", rc);
+
+            result([FlutterFFmpegPlugin toIntDictionary:KEY_RC :[NSNumber numberWithInt:rc]]);
+        });
+
+    } else if ([@"execute" isEqualToString:call.method]) {
+
+        if (delimiter == nil) {
+            delimiter = @" ";
+        }
+
+        NSLog(@"Running FFmpeg command: %@ with delimiter %@.\n", command, delimiter);
+
+        dispatch_async(dispatch_get_global_queue(DISPATCH_QUEUE_PRIORITY_DEFAULT, 0), ^{
+            int rc = [MobileFFmpeg execute:command delimiter:delimiter];
+
+            NSLog(@"FFmpeg exited with rc: %d\n", rc);
+
+            result([FlutterFFmpegPlugin toIntDictionary:KEY_RC :[NSNumber numberWithInt:rc]]);
+        });
+
+    } else if ([@"cancel" isEqualToString:call.method]) {
+
+        [MobileFFmpeg cancel];
+
+    } else if ([@"enableRedirection" isEqualToString:call.method]) {
+
+        [MobileFFmpegConfig enableRedirection];
+
+    } else if ([@"disableRedirection" isEqualToString:call.method]) {
+
+        [MobileFFmpegConfig disableRedirection];
+
+    } else if ([@"getLogLevel" isEqualToString:call.method]) {
+
+        int logLevel = [MobileFFmpegConfig getLogLevel];
+        result([FlutterFFmpegPlugin toIntDictionary:KEY_LOG_LEVEL :[NSNumber numberWithInt:logLevel]]);
+
+    } else if ([@"setLogLevel" isEqualToString:call.method]) {
+
+        NSNumber* logLevel = call.arguments[@"level"];
+        [MobileFFmpegConfig setLogLevel:[logLevel intValue]];
+
+    } else if ([@"enableLogs" isEqualToString:call.method]) {
+
+        [MobileFFmpegConfig setLogDelegate:self];
+
+    } else if ([@"disableLogs" isEqualToString:call.method]) {
+
+        [MobileFFmpegConfig setLogDelegate:nil];
+
+    } else if ([@"enableStatistics" isEqualToString:call.method]) {
+
+        [MobileFFmpegConfig setStatisticsDelegate:self];
+
+    } else if ([@"disableStatistics" isEqualToString:call.method]) {
+
+        [MobileFFmpegConfig setStatisticsDelegate:nil];
+
+    } else if ([@"getLastReceivedStatistics" isEqualToString:call.method]) {
+
+        Statistics *statistics = [MobileFFmpegConfig getLastReceivedStatistics];
+        result([FlutterFFmpegPlugin toStatisticsDictionary:statistics]);
+
+    } else if ([@"resetStatistics" isEqualToString:call.method]) {
+
+        [MobileFFmpegConfig resetStatistics];
+
+    } else if ([@"setFontconfigConfigurationPath" isEqualToString:call.method]) {
+
+        NSString* path = call.arguments[@"path"];
+        [MobileFFmpegConfig setFontconfigConfigurationPath:path];
+
+    } else if ([@"setFontDirectory" isEqualToString:call.method]) {
+
+        NSString* fontDirectoryPath = call.arguments[@"fontDirectory"];
+        NSDictionary* fontNameMapping = call.arguments[@"fontNameMap"];
+        [MobileFFmpegConfig setFontDirectory:fontDirectoryPath with:fontNameMapping];
+
+    } else if ([@"getPackageName" isEqualToString:call.method]) {
+
+        NSString *packageName = [MobileFFmpegConfig getPackageName];
+        result([FlutterFFmpegPlugin toStringDictionary:KEY_PACKAGE_NAME :packageName]);
+
+    } else if ([@"getExternalLibraries" isEqualToString:call.method]) {
+
+        NSArray *externalLibraries = [MobileFFmpegConfig getExternalLibraries];
+        result(externalLibraries);
+
+    } else if ([@"getLastReturnCode" isEqualToString:call.method]) {
+
+        int lastReturnCode = [MobileFFmpeg getLastReturnCode];
+        result([FlutterFFmpegPlugin toIntDictionary:KEY_LAST_RC :[NSNumber numberWithInt:lastReturnCode]]);
+
+    } else if ([@"getLastCommandOutput" isEqualToString:call.method]) {
+
+        NSString *lastCommandOutput = [MobileFFmpeg getLastCommandOutput];
+        result([FlutterFFmpegPlugin toStringDictionary:KEY_LAST_COMMAND_OUTPUT :lastCommandOutput]);
+
+    } else if ([@"getMediaInformation" isEqualToString:call.method]) {
+
+        NSString* path = call.arguments[@"path"];
+        NSNumber* timeout = call.arguments[@"timeout"];
+
+        NSLog(@"Getting media information for %@ with timeout %d.\n", path, [timeout intValue]);
+
+        dispatch_async(dispatch_get_global_queue(DISPATCH_QUEUE_PRIORITY_DEFAULT, 0), ^{
+            MediaInformation *mediaInformation = [MobileFFmpeg getMediaInformation:path timeout:[timeout intValue]];
+            result([FlutterFFmpegPlugin toMediaInformationDictionary:mediaInformation]);
+        });
+
+    } else {
+
+        result(FlutterMethodNotImplemented);
+
+    }
+}
+
+- (void)logCallback: (int)level :(NSString*)message {
+    dispatch_async(dispatch_get_main_queue(), ^{
+        NSMutableDictionary *dictionary = [[NSMutableDictionary alloc] init];
+        dictionary[KEY_LOG_LEVEL] = [NSNumber numberWithInt:level];
+        dictionary[KEY_LOG_TEXT] = message;
+
+        [self emitLogMessage: dictionary];
+    });
+}
+
+- (void)statisticsCallback:(Statistics *)statistics {
+    dispatch_async(dispatch_get_main_queue(), ^{
+        [self emitStatistics: statistics];
+    });
+}
+
+- (void)emitLogMessage:(NSDictionary*)logMessage{
+    _eventSink([FlutterFFmpegPlugin toStringDictionary:EVENT_LOG :logMessage]);
+}
+
+- (void)emitStatistics:(Statistics*)statistics{
+    NSDictionary *dictionary = [FlutterFFmpegPlugin toStatisticsDictionary:statistics];
+    _eventSink([FlutterFFmpegPlugin toStringDictionary:EVENT_STAT :dictionary]);
+}
+
++ (NSDictionary *)toStringDictionary:(NSString*)key :(NSString*)value {
+    NSMutableDictionary *dictionary = [[NSMutableDictionary alloc] init];
+    dictionary[key] = value;
+
+    return dictionary;
+}
+
++ (NSDictionary *)toIntDictionary:(NSString*)key :(NSNumber*)value {
+    NSMutableDictionary *dictionary = [[NSMutableDictionary alloc] init];
+    dictionary[key] = value;
+
+    return dictionary;
+}
+
++ (NSDictionary *)toStatisticsDictionary:(Statistics*)statistics {
+    NSMutableDictionary *dictionary = [[NSMutableDictionary alloc] init];
+
+    if (statistics != nil) {
+        dictionary[KEY_STAT_TIME] = [NSNumber numberWithInt: [statistics getTime]];
+        dictionary[KEY_STAT_SIZE] = [NSNumber numberWithLong: [statistics getSize]];
+
+        dictionary[KEY_STAT_BITRATE] = [NSNumber numberWithDouble: [statistics getBitrate]];
+        dictionary[KEY_STAT_SPEED] = [NSNumber numberWithDouble: [statistics getSpeed]];
+
+        dictionary[KEY_STAT_VIDEO_FRAME_NUMBER] = [NSNumber numberWithInt: [statistics getVideoFrameNumber]];
+        dictionary[KEY_STAT_VIDEO_QUALITY] = [NSNumber numberWithFloat: [statistics getVideoQuality]];
+        dictionary[KEY_STAT_VIDEO_FPS] = [NSNumber numberWithFloat: [statistics getVideoFps]];
+    }
+
+    return dictionary;
+}
+
++ (NSDictionary *)toMediaInformationDictionary:(MediaInformation*)mediaInformation {
+    NSMutableDictionary *dictionary = [[NSMutableDictionary alloc] init];
+
+    if (mediaInformation != nil) {
+        if ([mediaInformation getFormat] != nil) {
+            dictionary[@"format"] =  [mediaInformation getFormat];
+        }
+        if ([mediaInformation getPath] != nil) {
+            dictionary[@"path"] = [mediaInformation getPath];
+        }
+        if ([mediaInformation getStartTime] != nil) {
+            dictionary[@"startTime"] = [mediaInformation getStartTime];
+        }
+        if ([mediaInformation getDuration] != nil) {
+            dictionary[@"duration"] = [mediaInformation getDuration];
+        }
+        if ([mediaInformation getBitrate] != nil) {
+            dictionary[@"bitrate"] = [mediaInformation getBitrate];
+        }
+        if ([mediaInformation getRawInformation] != nil) {
+            dictionary[@"rawInformation"] = [mediaInformation getRawInformation];
+        }
+
+        NSDictionary *metadataDictionary = [mediaInformation getMetadataEntries];
+        if (metadataDictionary != nil && ([metadataDictionary count] > 0)) {
+            dictionary[@"metadata"] = metadataDictionary;
+        }
+
+        NSArray *streams = [mediaInformation getStreams];
+        if (streams != nil && ([streams count] > 0)) {
+            NSMutableArray *array = [[NSMutableArray alloc] init];
+
+            for (int i=0; i < [streams count]; i++) {
+                StreamInformation *streamInformation= [streams objectAtIndex:i];
+                [array addObject: [FlutterFFmpegPlugin toStreamInformationDictionary:streamInformation]];
+            }
+
+            dictionary[@"streams"] = array;
+        }
+    }
+
+    return dictionary;
+}
+
++ (NSDictionary *)toStreamInformationDictionary:(StreamInformation*)streamInformation {
+    NSMutableDictionary *dictionary = [[NSMutableDictionary alloc] init];
+
+    if (streamInformation != nil) {
+        if ([streamInformation getIndex] != nil) {
+            dictionary[@"index"] = [streamInformation getIndex];
+        }
+        if ([streamInformation getType] != nil) {
+            dictionary[@"type"] = [streamInformation getType];
+        }
+        if ([streamInformation getCodec] != nil) {
+            dictionary[@"codec"] = [streamInformation getCodec];
+        }
+        if ([streamInformation getFullCodec] != nil) {
+            dictionary[@"fullCodec"] = [streamInformation getFullCodec];
+        }
+        if ([streamInformation getFormat] != nil) {
+            dictionary[@"format"] = [streamInformation getFormat];
+        }
+        if ([streamInformation getFullFormat] != nil) {
+            dictionary[@"fullFormat"] = [streamInformation getFullFormat];
+        }
+        if ([streamInformation getWidth] != nil) {
+            dictionary[@"width"] = [streamInformation getWidth];
+        }
+        if ([streamInformation getHeight] != nil) {
+            dictionary[@"height"] = [streamInformation getHeight];
+        }
+        if ([streamInformation getBitrate] != nil) {
+            dictionary[@"bitrate"] = [streamInformation getBitrate];
+        }
+        if ([streamInformation getSampleRate] != nil) {
+            dictionary[@"sampleRate"] = [streamInformation getSampleRate];
+        }
+        if ([streamInformation getSampleFormat] != nil) {
+            dictionary[@"sampleFormat"] = [streamInformation getSampleFormat];
+        }
+        if ([streamInformation getChannelLayout] != nil) {
+            dictionary[@"channelLayout"] = [streamInformation getChannelLayout];
+        }
+        if ([streamInformation getSampleAspectRatio] != nil) {
+            dictionary[@"sampleAspectRatio"] = [streamInformation getSampleAspectRatio];
+        }
+        if ([streamInformation getDisplayAspectRatio] != nil) {
+            dictionary[@"displayAspectRatio"] = [streamInformation getDisplayAspectRatio];
+        }
+        if ([streamInformation getAverageFrameRate] != nil) {
+            dictionary[@"averageFrameRate"] = [streamInformation getAverageFrameRate];
+        }
+        if ([streamInformation getRealFrameRate] != nil) {
+            dictionary[@"realFrameRate"] = [streamInformation getRealFrameRate];
+        }
+        if ([streamInformation getTimeBase] != nil) {
+            dictionary[@"timeBase"] = [streamInformation getTimeBase];
+        }
+        if ([streamInformation getCodecTimeBase] != nil) {
+            dictionary[@"codecTimeBase"] = [streamInformation getCodecTimeBase];
+        }
+
+        NSDictionary *metadataDictionary = [streamInformation getMetadataEntries];
+        if (metadataDictionary != nil && ([metadataDictionary count] > 0)) {
+            dictionary[@"metadata"] = metadataDictionary;
+        }
+    }
+
+    return dictionary;
+}
+
+@end

+ 22 - 0
packages/flutter_ffmpeg_https-gpl/ios/flutter_ffmpeg.podspec

@@ -0,0 +1,22 @@
+Pod::Spec.new do |s|
+  s.name             = 'flutter_ffmpeg'
+  s.version          = '0.1.0'
+  s.summary          = 'FFmpeg plugin for Flutter.'
+  s.description      = 'FFmpeg plugin based on mobile-ffmpeg for Flutter.'
+  s.homepage         = 'https://github.com/tanersener/flutter-ffmpeg'
+
+  s.author           = { 'Taner Sener' => 'tanersener@gmail.com' }
+  s.license          = { :file => '../LICENSE' }
+
+  s.requires_arc     = true
+  s.ios.deployment_target = '9.3'
+
+  s.source              = { :path => '.' }
+  s.source_files        = 'Classes/**/*'
+  s.public_header_files = 'Classes/**/*.h'
+
+  s.dependency      'Flutter'
+  s.dependency      'mobile-ffmpeg-https-gpl', '4.2.LTS'
+
+end
+

+ 352 - 0
packages/flutter_ffmpeg_https-gpl/lib/flutter_ffmpeg.dart

@@ -0,0 +1,352 @@
+/*
+ * Copyright (c) 2019 Taner Sener
+ *
+ * This file is part of FlutterFFmpeg.
+ *
+ * FlutterFFmpeg is free software: you can redistribute it and/or modify
+ * it under the terms of the GNU Lesser General Public License as published by
+ * the Free Software Foundation, either version 3 of the License, or
+ * (at your option) any later version.
+ *
+ * FlutterFFmpeg is distributed in the hope that it will be useful,
+ * but WITHOUT ANY WARRANTY; without even the implied warranty of
+ * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
+ * GNU Lesser General Public License for more details.
+ *
+ * You should have received a copy of the GNU Lesser General Public License
+ * along with FlutterFFmpeg.  If not, see <http://www.gnu.org/licenses/>.
+ */
+
+import 'dart:async';
+
+import 'package:flutter/services.dart';
+
+class FlutterFFmpeg {
+  static const MethodChannel _methodChannel = const MethodChannel('flutter_ffmpeg');
+  static const EventChannel _eventChannel = const EventChannel('flutter_ffmpeg_event');
+
+  Function(int level, String message) logCallback;
+  Function(int time, int size, double bitrate, double speed, int videoFrameNumber, double videoQuality, double videoFps) statisticsCallback;
+
+  FlutterFFmpeg() {
+    logCallback = null;
+    statisticsCallback = null;
+
+    print("Loading flutter-ffmpeg.");
+
+    _eventChannel.receiveBroadcastStream().listen(_onEvent, onError: _onError);
+
+    enableLogs();
+    enableStatistics();
+    enableRedirection();
+
+    getPlatform().then((name) => print("Loaded flutter-ffmpeg-$name."));
+  }
+
+  void _onEvent(Object event) {
+    if (event is Map<dynamic, dynamic>) {
+      final Map<String, dynamic> eventMap = event.cast();
+      final Map<dynamic, dynamic> logEvent = eventMap['FlutterFFmpegLogCallback'];
+      final Map<dynamic, dynamic> statisticsEvent = eventMap['FlutterFFmpegStatisticsCallback'];
+
+      if (logEvent != null) {
+        int level = logEvent['level'];
+        String message = logEvent['log'];
+
+        if (this.logCallback == null) {
+          if (message.length > 0) {
+            // PRINT ALREADY ADDS NEW LINE. SO REMOVE THIS ONE
+            if (message.endsWith('\n')) {
+              print(message.substring(0, message.length - 1));
+            } else {
+              print(message);
+            }
+          }
+        } else {
+          this.logCallback(level, message);
+        }
+      }
+
+      if (statisticsEvent != null) {
+        if (this.statisticsCallback != null) {
+          int time = statisticsEvent['time'];
+          int size = statisticsEvent['size'];
+          double bitrate = _doublePrecision(statisticsEvent['bitrate'], 2);
+          double speed = _doublePrecision(statisticsEvent['speed'], 2);
+          int videoFrameNumber = statisticsEvent['videoFrameNumber'];
+          double videoQuality = _doublePrecision(statisticsEvent['videoQuality'], 2);
+          double videoFps = _doublePrecision(statisticsEvent['videoFps'], 2);
+
+          this.statisticsCallback(time, size, bitrate, speed, videoFrameNumber, videoQuality, videoFps);
+        }
+      }
+    }
+  }
+
+  void _onError(Object error) {
+    print('Event error: $error');
+  }
+
+  double _doublePrecision(double value, int precision) {
+    if (value == null) {
+      return 0;
+    } else {
+      return num.parse(value.toStringAsFixed(precision));
+    }
+  }
+
+  /// Returns FFmpeg version bundled within the library.
+  Future<String> getFFmpegVersion() async {
+    try {
+      final Map<dynamic, dynamic> result = await _methodChannel.invokeMethod('getFFmpegVersion');
+      return result['version'];
+    } on PlatformException catch (e) {
+      print("Plugin error: ${e.message}");
+      return null;
+    }
+  }
+
+  /// Returns platform name where library is loaded.
+  Future<String> getPlatform() async {
+    try {
+      final Map<dynamic, dynamic> result = await _methodChannel.invokeMethod('getPlatform');
+      return result['platform'];
+    } on PlatformException catch (e) {
+      print("Plugin error: ${e.message}");
+      return null;
+    }
+  }
+
+  /// Executes FFmpeg with [commandArguments] provided.
+  Future<int> executeWithArguments(List<String> arguments) async {
+    try {
+      final Map<dynamic, dynamic> result = await _methodChannel.invokeMethod('executeWithArguments', {'arguments': arguments});
+      return result['rc'];
+    } on PlatformException catch (e) {
+      print("Plugin error: ${e.message}");
+      return -1;
+    }
+  }
+
+  /// Executes FFmpeg [command] provided. Command is split into arguments using provided [delimiter].
+  Future<int> execute(String command, [String delimiter = ' ']) async {
+    try {
+      final Map<dynamic, dynamic> result = await _methodChannel.invokeMethod('execute', {'command': command, 'delimiter': delimiter});
+      return result['rc'];
+    } on PlatformException catch (e) {
+      print("Plugin error: ${e.message}");
+      return -1;
+    }
+  }
+
+  /// Cancels an ongoing operation.
+  Future<void> cancel() async {
+    try {
+      await _methodChannel.invokeMethod('cancel');
+    } on PlatformException catch (e) {
+      print("Plugin error: ${e.message}");
+    }
+  }
+
+  /// Enables redirection
+  Future<void> enableRedirection() async {
+    try {
+      await _methodChannel.invokeMethod('enableRedirection');
+    } on PlatformException catch (e) {
+      print("Plugin error: ${e.message}");
+    }
+  }
+
+  /// Disables log and statistics redirection. By default redirection is enabled in constructor.
+  /// When redirection is enabled FFmpeg logs are printed to console and can be routed further to a callback function.
+  /// By disabling redirection, logs are redirected to stderr.
+  /// Statistics redirection behaviour is similar. Statistics are not printed at all if redirection is not enabled.
+  /// If it is enabled then it is possible to define a statistics callback function but if you don't, they are not
+  /// printed anywhere and only saved as codelastReceivedStatistics data which can be polled with
+  /// [getLastReceivedStatistics()].
+  Future<void> disableRedirection() async {
+    try {
+      await _methodChannel.invokeMethod('disableRedirection');
+    } on PlatformException catch (e) {
+      print("Plugin error: ${e.message}");
+    }
+  }
+
+  /// Returns log level.
+  Future<int> getLogLevel() async {
+    try {
+      final Map<dynamic, dynamic> result = await _methodChannel.invokeMethod('getLogLevel');
+      return result['level'];
+    } on PlatformException catch (e) {
+      print("Plugin error: ${e.message}");
+      return -1;
+    }
+  }
+
+  /// Sets log level.
+  Future<void> setLogLevel(int logLevel) async {
+    try {
+      await _methodChannel.invokeMethod('setLogLevel', {'level': logLevel});
+    } on PlatformException catch (e) {
+      print("Plugin error: ${e.message}");
+    }
+  }
+
+  /// Enables log events
+  Future<void> enableLogs() async {
+    try {
+      await _methodChannel.invokeMethod('enableLogs');
+    } on PlatformException catch (e) {
+      print("Plugin error: ${e.message}");
+    }
+  }
+
+  /// Disables log functionality of the library. Logs will not be printed to console and log callback will be disabled.
+  /// Note that log functionality is enabled by default.
+  Future<void> disableLogs() async {
+    try {
+      await _methodChannel.invokeMethod('disableLogs');
+    } on PlatformException catch (e) {
+      print("Plugin error: ${e.message}");
+    }
+  }
+
+  /// Enables statistics events.
+  Future<void> enableStatistics() async {
+    try {
+      await _methodChannel.invokeMethod('enableStatistics');
+    } on PlatformException catch (e) {
+      print("Plugin error: ${e.message}");
+    }
+  }
+
+  /// Disables statistics functionality of the library. Statistics callback will be disabled but the last received
+  /// statistics data will be still available.
+  /// Note that statistics functionality is enabled by default.
+  Future<void> disableStatistics() async {
+    try {
+      await _methodChannel.invokeMethod('disableStatistics');
+    } on PlatformException catch (e) {
+      print("Plugin error: ${e.message}");
+    }
+  }
+
+  /// Sets a callback to redirect FFmpeg logs. [newCallback] is a new log callback function, use null to disable a previously defined callback
+  void enableLogCallback(Function(int level, String message) newCallback) {
+    try {
+      this.logCallback = newCallback;
+    } on PlatformException catch (e) {
+      print("Plugin error: ${e.message}");
+    }
+  }
+
+  /// Sets a callback to redirect FFmpeg statistics. [newCallback] is a new statistics callback function, use null to disable a previously defined callback
+  void enableStatisticsCallback(Function(int time, int size, double bitrate, double speed, int videoFrameNumber, double videoQuality, double videoFps) newCallback) {
+    try {
+      this.statisticsCallback = newCallback;
+    } on PlatformException catch (e) {
+      print("Plugin error: ${e.message}");
+    }
+  }
+
+  /// Returns the last received statistics data stored in bitrate, size, speed, time, videoFps, videoFrameNumber and
+  /// videoQuality fields
+  Future<Map<dynamic, dynamic>> getLastReceivedStatistics() async {
+    try {
+      final Map<dynamic, dynamic> result = await _methodChannel.invokeMethod('getLastReceivedStatistics');
+      return result;
+    } on PlatformException catch (e) {
+      print("Plugin error: ${e.message}");
+      return null;
+    }
+  }
+
+  /// Resets last received statistics. It is recommended to call it before starting a new execution.
+  Future<void> resetStatistics() async {
+    try {
+      await _methodChannel.invokeMethod('resetStatistics');
+    } on PlatformException catch (e) {
+      print("Plugin error: ${e.message}");
+    }
+  }
+
+  /// Sets and overrides fontconfig configuration directory.
+  Future<void> setFontconfigConfigurationPath(String path) async {
+    try {
+      await _methodChannel.invokeMethod('setFontconfigConfigurationPath', {'path': path});
+    } on PlatformException catch (e) {
+      print("Plugin error: ${e.message}");
+    }
+  }
+
+  /// Registers fonts inside the given [fontDirectory], so they are available to use in FFmpeg filters.
+  Future<void> setFontDirectory(String fontDirectory, Map<String, String> fontNameMap) async {
+    var parameters;
+    if (fontNameMap == null) {
+      parameters = {'fontDirectory': fontDirectory};
+    } else {
+      parameters = {'fontDirectory': fontDirectory, 'fontNameMap': fontNameMap};
+    }
+
+    try {
+      await _methodChannel.invokeMethod('setFontDirectory', parameters);
+    } on PlatformException catch (e) {
+      print("Plugin error: ${e.message}");
+    }
+  }
+
+  /// Returns FlutterFFmpeg package name.
+  Future<String> getPackageName() async {
+    try {
+      final Map<dynamic, dynamic> result = await _methodChannel.invokeMethod('getPackageName');
+      return result['packageName'];
+    } on PlatformException catch (e) {
+      print("Plugin error: ${e.message}");
+      return null;
+    }
+  }
+
+  /// Returns supported external libraries.
+  Future<List<dynamic>> getExternalLibraries() async {
+    try {
+      final List<dynamic> result = await _methodChannel.invokeMethod('getExternalLibraries');
+      return result;
+    } on PlatformException catch (e) {
+      print("Plugin error: ${e.message}");
+      return null;
+    }
+  }
+
+  /// Returns return code of last executed command.
+  Future<int> getLastReturnCode() async {
+    try {
+      final Map<dynamic, dynamic> result = await _methodChannel.invokeMethod('getLastReturnCode');
+      return result['lastRc'];
+    } on PlatformException catch (e) {
+      print("Plugin error: ${e.message}");
+      return -1;
+    }
+  }
+
+  /// Returns log output of last executed command. Please note that disabling redirection using
+  /// [disableRedirection()] method also disables this functionality.
+  Future<String> getLastCommandOutput() async {
+    try {
+      final Map<dynamic, dynamic> result = await _methodChannel.invokeMethod('getLastCommandOutput');
+      return result['lastCommandOutput'];
+    } on PlatformException catch (e) {
+      print("Plugin error: ${e.message}");
+      return null;
+    }
+  }
+
+  /// Returns media information for given [path] using optional [timeout]
+  Future<Map<dynamic, dynamic>> getMediaInformation(String path, [int timeout = 10000]) async {
+    try {
+      return await _methodChannel.invokeMethod('getMediaInformation', {'path': path, 'timeout': timeout});
+    } on PlatformException catch (e) {
+      print("Plugin error: ${e.message}");
+      return null;
+    }
+  }
+}

+ 76 - 0
packages/flutter_ffmpeg_https-gpl/lib/log_level.dart

@@ -0,0 +1,76 @@
+/*
+ * Copyright (c) 2019 Taner Sener
+ *
+ * This file is part of FlutterFFmpeg.
+ *
+ * FlutterFFmpeg is free software: you can redistribute it and/or modify
+ * it under the terms of the GNU Lesser General Public License as published by
+ * the Free Software Foundation, either version 3 of the License, or
+ * (at your option) any later version.
+ *
+ * FlutterFFmpeg is distributed in the hope that it will be useful,
+ * but WITHOUT ANY WARRANTY; without even the implied warranty of
+ * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
+ * GNU Lesser General Public License for more details.
+ *
+ * You should have received a copy of the GNU Lesser General Public License
+ * along with FlutterFFmpeg.  If not, see <http://www.gnu.org/licenses/>.
+ */
+
+class LogLevel {
+  /// Print no output.
+  static const int AV_LOG_QUIET = -8;
+
+  /// Something went really wrong and we will crash now.
+  static const int AV_LOG_PANIC = 0;
+
+  /// Something went wrong and recovery is not possible.
+  /// For example, no header was found for a format which depends
+  /// on headers or an illegal combination of parameters is used.
+  static const int AV_LOG_FATAL = 8;
+
+  /// Something went wrong and cannot losslessly be recovered.
+  /// However, not all future data is affected.
+  static const int AV_LOG_ERROR = 16;
+
+  /// Something somehow does not look correct. This may or may not
+  /// lead to problems. An example would be the use of '-vstrict -2'.
+  static const int AV_LOG_WARNING = 24;
+
+  /// int Standard information.
+  static const int AV_LOG_INFO = 32;
+
+  /// Detailed information.
+  static const int AV_LOG_VERBOSE = 40;
+
+  /// Stuff which is only useful for libav* developers.
+  static const int AV_LOG_DEBUG = 48;
+
+  /// Extremely verbose debugging, useful for libav* development.
+  static const int AV_LOG_TRACE = 56;
+
+  /// Returns log level string from int
+  static String levelToString(int level) {
+    switch (level) {
+      case LogLevel.AV_LOG_TRACE:
+        return "TRACE";
+      case LogLevel.AV_LOG_DEBUG:
+        return "DEBUG";
+      case LogLevel.AV_LOG_VERBOSE:
+        return "VERBOSE";
+      case LogLevel.AV_LOG_INFO:
+        return "INFO";
+      case LogLevel.AV_LOG_WARNING:
+        return "WARNING";
+      case LogLevel.AV_LOG_ERROR:
+        return "ERROR";
+      case LogLevel.AV_LOG_FATAL:
+        return "FATAL";
+      case LogLevel.AV_LOG_PANIC:
+        return "PANIC";
+      case LogLevel.AV_LOG_QUIET:
+      default:
+        return "";
+    }
+  }
+}

+ 17 - 0
packages/flutter_ffmpeg_https-gpl/pubspec.yaml

@@ -0,0 +1,17 @@
+name: flutter_ffmpeg
+description: FFmpeg plugin for Flutter.
+version: 0.1.0
+author: Taner Sener <tanersener@gmail.com>
+homepage: https://github.com/tanersener/flutter-ffmpeg
+
+environment:
+  sdk: ">=2.0.0-dev.68.0 <3.0.0"
+
+dependencies:
+  flutter:
+    sdk: flutter
+
+flutter:
+  plugin:
+    androidPackage: com.arthenica.flutter.ffmpeg
+    pluginClass: FlutterFFmpegPlugin

+ 7 - 0
packages/flutter_ffmpeg_https/.gitignore

@@ -0,0 +1,7 @@
+.idea/
+.DS_Store
+.dart_tool/
+.pub/
+build/
+pubspec.lock
+.packages

+ 10 - 0
packages/flutter_ffmpeg_https/.metadata

@@ -0,0 +1,10 @@
+# This file tracks properties of this Flutter project.
+# Used by Flutter tool to assess capabilities and perform upgrades etc.
+#
+# This file should be version controlled and should not be manually edited.
+
+version:
+  revision: 5391447fae6209bb21a89e6a5a6583cac1af9b4b
+  channel: stable
+
+project_type: plugin

+ 165 - 0
packages/flutter_ffmpeg_https/LICENSE

@@ -0,0 +1,165 @@
+                   GNU LESSER GENERAL PUBLIC LICENSE
+                       Version 3, 29 June 2007
+
+ Copyright (C) 2007 Free Software Foundation, Inc. <http://fsf.org/>
+ Everyone is permitted to copy and distribute verbatim copies
+ of this license document, but changing it is not allowed.
+
+
+  This version of the GNU Lesser General Public License incorporates
+the terms and conditions of version 3 of the GNU General Public
+License, supplemented by the additional permissions listed below.
+
+  0. Additional Definitions.
+
+  As used herein, "this License" refers to version 3 of the GNU Lesser
+General Public License, and the "GNU GPL" refers to version 3 of the GNU
+General Public License.
+
+  "The Library" refers to a covered work governed by this License,
+other than an Application or a Combined Work as defined below.
+
+  An "Application" is any work that makes use of an interface provided
+by the Library, but which is not otherwise based on the Library.
+Defining a subclass of a class defined by the Library is deemed a mode
+of using an interface provided by the Library.
+
+  A "Combined Work" is a work produced by combining or linking an
+Application with the Library.  The particular version of the Library
+with which the Combined Work was made is also called the "Linked
+Version".
+
+  The "Minimal Corresponding Source" for a Combined Work means the
+Corresponding Source for the Combined Work, excluding any source code
+for portions of the Combined Work that, considered in isolation, are
+based on the Application, and not on the Linked Version.
+
+  The "Corresponding Application Code" for a Combined Work means the
+object code and/or source code for the Application, including any data
+and utility programs needed for reproducing the Combined Work from the
+Application, but excluding the System Libraries of the Combined Work.
+
+  1. Exception to Section 3 of the GNU GPL.
+
+  You may convey a covered work under sections 3 and 4 of this License
+without being bound by section 3 of the GNU GPL.
+
+  2. Conveying Modified Versions.
+
+  If you modify a copy of the Library, and, in your modifications, a
+facility refers to a function or data to be supplied by an Application
+that uses the facility (other than as an argument passed when the
+facility is invoked), then you may convey a copy of the modified
+version:
+
+   a) under this License, provided that you make a good faith effort to
+   ensure that, in the event an Application does not supply the
+   function or data, the facility still operates, and performs
+   whatever part of its purpose remains meaningful, or
+
+   b) under the GNU GPL, with none of the additional permissions of
+   this License applicable to that copy.
+
+  3. Object Code Incorporating Material from Library Header Files.
+
+  The object code form of an Application may incorporate material from
+a header file that is part of the Library.  You may convey such object
+code under terms of your choice, provided that, if the incorporated
+material is not limited to numerical parameters, data structure
+layouts and accessors, or small macros, inline functions and templates
+(ten or fewer lines in length), you do both of the following:
+
+   a) Give prominent notice with each copy of the object code that the
+   Library is used in it and that the Library and its use are
+   covered by this License.
+
+   b) Accompany the object code with a copy of the GNU GPL and this license
+   document.
+
+  4. Combined Works.
+
+  You may convey a Combined Work under terms of your choice that,
+taken together, effectively do not restrict modification of the
+portions of the Library contained in the Combined Work and reverse
+engineering for debugging such modifications, if you also do each of
+the following:
+
+   a) Give prominent notice with each copy of the Combined Work that
+   the Library is used in it and that the Library and its use are
+   covered by this License.
+
+   b) Accompany the Combined Work with a copy of the GNU GPL and this license
+   document.
+
+   c) For a Combined Work that displays copyright notices during
+   execution, include the copyright notice for the Library among
+   these notices, as well as a reference directing the user to the
+   copies of the GNU GPL and this license document.
+
+   d) Do one of the following:
+
+       0) Convey the Minimal Corresponding Source under the terms of this
+       License, and the Corresponding Application Code in a form
+       suitable for, and under terms that permit, the user to
+       recombine or relink the Application with a modified version of
+       the Linked Version to produce a modified Combined Work, in the
+       manner specified by section 6 of the GNU GPL for conveying
+       Corresponding Source.
+
+       1) Use a suitable shared library mechanism for linking with the
+       Library.  A suitable mechanism is one that (a) uses at run time
+       a copy of the Library already present on the user's computer
+       system, and (b) will operate properly with a modified version
+       of the Library that is interface-compatible with the Linked
+       Version.
+
+   e) Provide Installation Information, but only if you would otherwise
+   be required to provide such information under section 6 of the
+   GNU GPL, and only to the extent that such information is
+   necessary to install and execute a modified version of the
+   Combined Work produced by recombining or relinking the
+   Application with a modified version of the Linked Version. (If
+   you use option 4d0, the Installation Information must accompany
+   the Minimal Corresponding Source and Corresponding Application
+   Code. If you use option 4d1, you must provide the Installation
+   Information in the manner specified by section 6 of the GNU GPL
+   for conveying Corresponding Source.)
+
+  5. Combined Libraries.
+
+  You may place library facilities that are a work based on the
+Library side by side in a single library together with other library
+facilities that are not Applications and are not covered by this
+License, and convey such a combined library under terms of your
+choice, if you do both of the following:
+
+   a) Accompany the combined library with a copy of the same work based
+   on the Library, uncombined with any other library facilities,
+   conveyed under the terms of this License.
+
+   b) Give prominent notice with the combined library that part of it
+   is a work based on the Library, and explaining where to find the
+   accompanying uncombined form of the same work.
+
+  6. Revised Versions of the GNU Lesser General Public License.
+
+  The Free Software Foundation may publish revised and/or new versions
+of the GNU Lesser General Public License from time to time. Such new
+versions will be similar in spirit to the present version, but may
+differ in detail to address new problems or concerns.
+
+  Each version is given a distinguishing version number. If the
+Library as you received it specifies that a certain numbered version
+of the GNU Lesser General Public License "or any later version"
+applies to it, you have the option of following the terms and
+conditions either of that published version or of any later version
+published by the Free Software Foundation. If the Library as you
+received it does not specify a version number of the GNU Lesser
+General Public License, you may choose any version of the GNU Lesser
+General Public License ever published by the Free Software Foundation.
+
+  If the Library as you received it specifies that a proxy can decide
+whether future versions of the GNU Lesser General Public License shall
+apply, that proxy's public statement of acceptance of any version is
+permanent authorization for you to choose that version for the
+Library.

+ 8 - 0
packages/flutter_ffmpeg_https/android/.gitignore

@@ -0,0 +1,8 @@
+*.iml
+.gradle
+/local.properties
+/.idea/workspace.xml
+/.idea/libraries
+.DS_Store
+/build
+/captures

+ 38 - 0
packages/flutter_ffmpeg_https/android/build.gradle

@@ -0,0 +1,38 @@
+group 'com.arthenica.flutter.ffmpeg'
+version '0.1.0'
+
+buildscript {
+    repositories {
+        google()
+        jcenter()
+    }
+
+    dependencies {
+        classpath 'com.android.tools.build:gradle:3.2.1'
+    }
+}
+
+rootProject.allprojects {
+    repositories {
+        google()
+        jcenter()
+    }
+}
+
+apply plugin: 'com.android.library'
+
+android {
+    compileSdkVersion 27
+
+    defaultConfig {
+        minSdkVersion 21
+        testInstrumentationRunner "android.support.test.runner.AndroidJUnitRunner"
+    }
+    lintOptions {
+        disable 'InvalidPackage'
+    }
+}
+
+dependencies {
+    implementation 'com.arthenica:mobile-ffmpeg-https:4.2.LTS'
+}

+ 1 - 0
packages/flutter_ffmpeg_https/android/gradle.properties

@@ -0,0 +1 @@
+org.gradle.jvmargs=-Xmx1536M

+ 1 - 0
packages/flutter_ffmpeg_https/android/settings.gradle

@@ -0,0 +1 @@
+rootProject.name = 'flutter_ffmpeg'

+ 3 - 0
packages/flutter_ffmpeg_https/android/src/main/AndroidManifest.xml

@@ -0,0 +1,3 @@
+<manifest xmlns:android="http://schemas.android.com/apk/res/android"
+  package="com.arthenica.flutter.ffmpeg">
+</manifest>

+ 66 - 0
packages/flutter_ffmpeg_https/android/src/main/java/com/arthenica/flutter/ffmpeg/FlutterFFmpegExecuteAsyncArgumentsTask.java

@@ -0,0 +1,66 @@
+/*
+ * Copyright (c) 2019 Taner Sener
+ *
+ * This file is part of FlutterFFmpeg.
+ *
+ * FlutterFFmpeg is free software: you can redistribute it and/or modify
+ * it under the terms of the GNU Lesser General Public License as published by
+ * the Free Software Foundation, either version 3 of the License, or
+ * (at your option) any later version.
+ *
+ * FlutterFFmpeg is distributed in the hope that it will be useful,
+ * but WITHOUT ANY WARRANTY; without even the implied warranty of
+ * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
+ * GNU Lesser General Public License for more details.
+ *
+ * You should have received a copy of the GNU Lesser General Public License
+ * along with FlutterFFmpeg.  If not, see <http://www.gnu.org/licenses/>.
+ */
+
+package com.arthenica.flutter.ffmpeg;
+
+import android.os.AsyncTask;
+import android.util.Log;
+
+import com.arthenica.mobileffmpeg.FFmpeg;
+
+import java.util.Arrays;
+import java.util.List;
+
+import io.flutter.plugin.common.MethodChannel;
+
+/**
+ * Asynchronous task which performs {@link FFmpeg#execute(String[])} method invocations.
+ *
+ * @author Taner Sener
+ * @since 0.1.0
+ */
+public class FlutterFFmpegExecuteAsyncArgumentsTask extends AsyncTask<String, Integer, Integer> {
+
+    private final MethodChannel.Result result;
+    private final List<String> arguments;
+
+    FlutterFFmpegExecuteAsyncArgumentsTask(final List<String> arguments, final MethodChannel.Result result) {
+        this.arguments = arguments;
+        this.result = result;
+    }
+
+    @Override
+    protected Integer doInBackground(final String... dummyString) {
+        final String[] argumentsArray = arguments.toArray(new String[0]);
+
+        Log.d(FlutterFFmpegPlugin.LIBRARY_NAME, String.format("Running FFmpeg with arguments: %s.", Arrays.toString(argumentsArray)));
+
+        int rc = FFmpeg.execute(argumentsArray);
+
+        Log.d(FlutterFFmpegPlugin.LIBRARY_NAME, String.format("FFmpeg exited with rc: %d", rc));
+
+        return rc;
+    }
+
+    @Override
+    protected void onPostExecute(final Integer rc) {
+        result.success(FlutterFFmpegPlugin.toIntMap(FlutterFFmpegPlugin.KEY_RC, rc));
+    }
+
+}

+ 72 - 0
packages/flutter_ffmpeg_https/android/src/main/java/com/arthenica/flutter/ffmpeg/FlutterFFmpegExecuteAsyncCommandTask.java

@@ -0,0 +1,72 @@
+/*
+ * Copyright (c) 2019 Taner Sener
+ *
+ * This file is part of FlutterFFmpeg.
+ *
+ * FlutterFFmpeg is free software: you can redistribute it and/or modify
+ * it under the terms of the GNU Lesser General Public License as published by
+ * the Free Software Foundation, either version 3 of the License, or
+ * (at your option) any later version.
+ *
+ * FlutterFFmpeg is distributed in the hope that it will be useful,
+ * but WITHOUT ANY WARRANTY; without even the implied warranty of
+ * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
+ * GNU Lesser General Public License for more details.
+ *
+ * You should have received a copy of the GNU Lesser General Public License
+ * along with FlutterFFmpeg.  If not, see <http://www.gnu.org/licenses/>.
+ */
+
+package com.arthenica.flutter.ffmpeg;
+
+import android.os.AsyncTask;
+import android.util.Log;
+
+import com.arthenica.mobileffmpeg.FFmpeg;
+
+import io.flutter.plugin.common.MethodChannel;
+
+/**
+ * Asynchronous task which performs {@link FFmpeg#execute(String, String)} method invocations.
+ *
+ * @author Taner Sener
+ * @since 0.1.0
+ */
+public class FlutterFFmpegExecuteAsyncCommandTask extends AsyncTask<String, Integer, Integer> {
+
+    private String delimiter;
+    private final MethodChannel.Result result;
+
+    FlutterFFmpegExecuteAsyncCommandTask(final String delimiter, final MethodChannel.Result result) {
+        if (delimiter == null) {
+            this.delimiter = " ";
+        } else {
+            this.delimiter = delimiter;
+        }
+
+        this.result = result;
+    }
+
+    @Override
+    protected Integer doInBackground(final String... strings) {
+        int rc = -1;
+
+        if ((strings != null) && (strings.length > 0)) {
+            final String command = strings[0];
+
+            Log.d(FlutterFFmpegPlugin.LIBRARY_NAME, String.format("Running FFmpeg command: %s with delimiter %s.", command, delimiter));
+
+            rc = FFmpeg.execute(command, delimiter);
+
+            Log.d(FlutterFFmpegPlugin.LIBRARY_NAME, String.format("FFmpeg exited with rc: %d", rc));
+        }
+
+        return rc;
+    }
+
+    @Override
+    protected void onPostExecute(final Integer rc) {
+        result.success(FlutterFFmpegPlugin.toIntMap(FlutterFFmpegPlugin.KEY_RC, rc));
+    }
+
+}

+ 70 - 0
packages/flutter_ffmpeg_https/android/src/main/java/com/arthenica/flutter/ffmpeg/FlutterFFmpegGetMediaInformationAsyncTask.java

@@ -0,0 +1,70 @@
+/*
+ * Copyright (c) 2019 Taner Sener
+ *
+ * This file is part of FlutterFFmpeg.
+ *
+ * FlutterFFmpeg is free software: you can redistribute it and/or modify
+ * it under the terms of the GNU Lesser General Public License as published by
+ * the Free Software Foundation, either version 3 of the License, or
+ * (at your option) any later version.
+ *
+ * FlutterFFmpeg is distributed in the hope that it will be useful,
+ * but WITHOUT ANY WARRANTY; without even the implied warranty of
+ * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
+ * GNU Lesser General Public License for more details.
+ *
+ * You should have received a copy of the GNU Lesser General Public License
+ * along with FlutterFFmpeg.  If not, see <http://www.gnu.org/licenses/>.
+ */
+
+package com.arthenica.flutter.ffmpeg;
+
+import android.os.AsyncTask;
+import android.util.Log;
+
+import com.arthenica.mobileffmpeg.FFmpeg;
+import com.arthenica.mobileffmpeg.MediaInformation;
+
+import io.flutter.plugin.common.MethodChannel;
+
+/**
+ * Asynchronous task which performs {@link FFmpeg#getMediaInformation(String, Long)} method invocations.
+ *
+ * @author Taner Sener
+ * @since 0.1.0
+ */
+public class FlutterFFmpegGetMediaInformationAsyncTask extends AsyncTask<String, Integer, MediaInformation> {
+
+    private Integer timeout;
+    private final MethodChannel.Result result;
+
+    FlutterFFmpegGetMediaInformationAsyncTask(final Integer timeout, final MethodChannel.Result result) {
+        this.timeout = timeout;
+        this.result = result;
+    }
+
+    @Override
+    protected MediaInformation doInBackground(final String... strings) {
+        MediaInformation mediaInformation = null;
+
+        if ((strings != null) && (strings.length > 0)) {
+            final String path = strings[0];
+
+            if (timeout == null) {
+                Log.d(FlutterFFmpegPlugin.LIBRARY_NAME, String.format("Getting media information for %s", path));
+                mediaInformation = FFmpeg.getMediaInformation(path);
+            } else {
+                Log.d(FlutterFFmpegPlugin.LIBRARY_NAME, String.format("Getting media information for %s with timeout %d.", path, timeout.longValue()));
+                mediaInformation = FFmpeg.getMediaInformation(path, timeout.longValue());
+            }
+        }
+
+        return mediaInformation;
+    }
+
+    @Override
+    protected void onPostExecute(final MediaInformation mediaInformation) {
+        result.success(FlutterFFmpegPlugin.toMediaInformationMap(mediaInformation));
+    }
+
+}

+ 436 - 0
packages/flutter_ffmpeg_https/android/src/main/java/com/arthenica/flutter/ffmpeg/FlutterFFmpegPlugin.java

@@ -0,0 +1,436 @@
+/*
+ * Copyright (c) 2019 Taner Sener
+ *
+ * This file is part of FlutterFFmpeg.
+ *
+ * FlutterFFmpeg is free software: you can redistribute it and/or modify
+ * it under the terms of the GNU Lesser General Public License as published by
+ * the Free Software Foundation, either version 3 of the License, or
+ * (at your option) any later version.
+ *
+ * FlutterFFmpeg is distributed in the hope that it will be useful,
+ * but WITHOUT ANY WARRANTY; without even the implied warranty of
+ * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
+ * GNU Lesser General Public License for more details.
+ *
+ * You should have received a copy of the GNU Lesser General Public License
+ * along with FlutterFFmpeg.  If not, see <http://www.gnu.org/licenses/>.
+ */
+
+package com.arthenica.flutter.ffmpeg;
+
+import android.content.Context;
+import android.system.ErrnoException;
+import android.util.Log;
+
+import com.arthenica.mobileffmpeg.AbiDetect;
+import com.arthenica.mobileffmpeg.Config;
+import com.arthenica.mobileffmpeg.FFmpeg;
+import com.arthenica.mobileffmpeg.Level;
+import com.arthenica.mobileffmpeg.LogCallback;
+import com.arthenica.mobileffmpeg.LogMessage;
+import com.arthenica.mobileffmpeg.MediaInformation;
+import com.arthenica.mobileffmpeg.Statistics;
+import com.arthenica.mobileffmpeg.StatisticsCallback;
+import com.arthenica.mobileffmpeg.StreamInformation;
+
+import java.util.ArrayList;
+import java.util.HashMap;
+import java.util.List;
+import java.util.Map;
+import java.util.Set;
+
+import io.flutter.plugin.common.EventChannel;
+import io.flutter.plugin.common.MethodCall;
+import io.flutter.plugin.common.MethodChannel;
+import io.flutter.plugin.common.MethodChannel.MethodCallHandler;
+import io.flutter.plugin.common.MethodChannel.Result;
+import io.flutter.plugin.common.PluginRegistry.Registrar;
+
+/**
+ * <h3>Flutter FFmpeg Plugin</h3>
+ *
+ * @author Taner Sener
+ * @since 0.1.0
+ */
+public class FlutterFFmpegPlugin implements MethodCallHandler, EventChannel.StreamHandler {
+    public static final String LIBRARY_NAME = "flutter-ffmpeg";
+
+    public static final String PLATFORM_NAME = "android";
+    public static final String KEY_VERSION = "version";
+    public static final String KEY_RC = "rc";
+    public static final String KEY_PLATFORM = "platform";
+    public static final String KEY_PACKAGE_NAME = "packageName";
+    public static final String KEY_LAST_RC = "lastRc";
+
+    public static final String KEY_LAST_COMMAND_OUTPUT = "lastCommandOutput";
+    public static final String KEY_LOG_TEXT = "log";
+
+    public static final String KEY_LOG_LEVEL = "level";
+    public static final String KEY_STAT_TIME = "time";
+    public static final String KEY_STAT_SIZE = "size";
+    public static final String KEY_STAT_BITRATE = "bitrate";
+    public static final String KEY_STAT_SPEED = "speed";
+    public static final String KEY_STAT_VIDEO_FRAME_NUMBER = "videoFrameNumber";
+    public static final String KEY_STAT_VIDEO_QUALITY = "videoQuality";
+    public static final String KEY_STAT_VIDEO_FPS = "videoFps";
+
+    public static final String EVENT_LOG = "FlutterFFmpegLogCallback";
+    public static final String EVENT_STAT = "FlutterFFmpegStatisticsCallback";
+
+    private EventChannel.EventSink eventSink;
+    private final Registrar registrar;
+
+    /**
+     * Registers plugin to registry.
+     *
+     * @param registrar receiver of plugin registration
+     */
+    public static void registerWith(final Registrar registrar) {
+        FlutterFFmpegPlugin handler = new FlutterFFmpegPlugin(registrar);
+
+        final MethodChannel channel = new MethodChannel(registrar.messenger(), "flutter_ffmpeg");
+        channel.setMethodCallHandler(handler);
+
+        final EventChannel eventChannel = new EventChannel(registrar.messenger(), "flutter_ffmpeg_event");
+        eventChannel.setStreamHandler(handler);
+    }
+
+    private FlutterFFmpegPlugin(Registrar registrar) {
+        this.registrar = registrar;
+    }
+
+    private Context getActiveContext() {
+        return (registrar.activity() != null) ? registrar.activity() : registrar.context();
+    }
+
+    /**
+     * Handles method calls.
+     *
+     * @param call   method call
+     * @param result result callback
+     */
+    @Override
+    public void onMethodCall(final MethodCall call, final Result result) {
+        if (call.method.equals("getPlatform")) {
+
+            final String abi = AbiDetect.getAbi();
+            result.success(toStringMap(KEY_PLATFORM, PLATFORM_NAME + "-" + abi));
+
+        } else if (call.method.equals("getFFmpegVersion")) {
+
+            final String version = FFmpeg.getFFmpegVersion();
+            result.success(toStringMap(KEY_VERSION, version));
+
+        } else if (call.method.equals("executeWithArguments")) {
+
+            List<String> arguments = call.argument("arguments");
+
+            final FlutterFFmpegExecuteAsyncArgumentsTask asyncTask = new FlutterFFmpegExecuteAsyncArgumentsTask(arguments, result);
+            asyncTask.execute("dummy-trigger");
+
+        } else if (call.method.equals("execute")) {
+
+            String command = call.argument("command");
+            String delimiter = call.argument("delimiter");
+
+            final FlutterFFmpegExecuteAsyncCommandTask asyncTask = new FlutterFFmpegExecuteAsyncCommandTask(delimiter, result);
+            asyncTask.execute(command);
+
+        } else if (call.method.equals("cancel")) {
+
+            FFmpeg.cancel();
+
+        } else if (call.method.equals("enableRedirection")) {
+
+            Config.enableRedirection();
+
+        } else if (call.method.equals("disableRedirection")) {
+
+            Config.disableRedirection();
+
+        } else if (call.method.equals("getLogLevel")) {
+
+            final Level level = Config.getLogLevel();
+            result.success(toIntMap(KEY_LOG_LEVEL, levelToInt(level)));
+
+        } else if (call.method.equals("setLogLevel")) {
+
+            Integer level = call.argument("level");
+            if (level == null) {
+                level = Level.AV_LOG_TRACE.getValue();
+            }
+            Config.setLogLevel(Level.from(level));
+
+        } else if (call.method.equals("enableLogs")) {
+
+            Config.enableLogCallback(new LogCallback() {
+
+                @Override
+                public void apply(final LogMessage logMessage) {
+                    emitLogMessage(logMessage);
+                }
+            });
+
+        } else if (call.method.equals("disableLogs")) {
+
+            Config.enableLogCallback(null);
+
+        } else if (call.method.equals("enableStatistics")) {
+
+            Config.enableStatisticsCallback(new StatisticsCallback() {
+
+                @Override
+                public void apply(final Statistics statistics) {
+                    emitStatistics(statistics);
+                }
+            });
+
+        } else if (call.method.equals("disableStatistics")) {
+
+            Config.enableStatisticsCallback(null);
+
+        } else if (call.method.equals("getLastReceivedStatistics")) {
+
+            result.success(toMap(Config.getLastReceivedStatistics()));
+
+        } else if (call.method.equals("resetStatistics")) {
+
+            Config.resetStatistics();
+
+        } else if (call.method.equals("setFontconfigConfigurationPath")) {
+            String path = call.argument("path");
+
+            try {
+                Config.setFontconfigConfigurationPath(path);
+            } catch (final ErrnoException e) {
+                Log.w(LIBRARY_NAME, String.format("Setting fontconfig configuration path failed for %s", path), e);
+            }
+
+        } else if (call.method.equals("setFontDirectory")) {
+
+            String path = call.argument("fontDirectory");
+            Map<String, String> map = call.argument("fontNameMap");
+
+            Config.setFontDirectory(getActiveContext(), path, map);
+
+        } else if (call.method.equals("getPackageName")) {
+
+            final String packageName = Config.getPackageName();
+            result.success(toStringMap(KEY_PACKAGE_NAME, packageName));
+
+        } else if (call.method.equals("getExternalLibraries")) {
+
+            final List<String> externalLibraries = Config.getExternalLibraries();
+            result.success(externalLibraries);
+
+        } else if (call.method.equals("getLastReturnCode")) {
+
+            int lastReturnCode = FFmpeg.getLastReturnCode();
+            result.success(toIntMap(KEY_LAST_RC, lastReturnCode));
+
+        } else if (call.method.equals("getLastCommandOutput")) {
+
+            final String lastCommandOutput = FFmpeg.getLastCommandOutput();
+            result.success(toStringMap(KEY_LAST_COMMAND_OUTPUT, lastCommandOutput));
+
+        } else if (call.method.equals("getMediaInformation")) {
+            final String path = call.argument("path");
+            Integer timeout = call.argument("timeout");
+            if (timeout == null) {
+                timeout = 10000;
+            }
+
+            final FlutterFFmpegGetMediaInformationAsyncTask asyncTask = new FlutterFFmpegGetMediaInformationAsyncTask(timeout, result);
+            asyncTask.execute(path);
+
+        } else {
+            result.notImplemented();
+        }
+    }
+
+    @Override
+    public void onListen(Object o, EventChannel.EventSink eventSink) {
+        this.eventSink = eventSink;
+    }
+
+    @Override
+    public void onCancel(Object o) {
+        this.eventSink = null;
+    }
+
+    protected void emitLogMessage(final LogMessage logMessage) {
+        final HashMap<String, Object> logWrapperMap = new HashMap<>();
+        final HashMap<String, Object> logMap = new HashMap<>();
+
+        logMap.put(KEY_LOG_LEVEL, levelToInt(logMessage.getLevel()));
+        logMap.put(KEY_LOG_TEXT, logMessage.getText());
+
+        logWrapperMap.put(EVENT_LOG, logMap);
+
+        eventSink.success(logWrapperMap);
+    }
+
+    protected void emitStatistics(final Statistics statistics) {
+        final HashMap<String, Object> statisticsMap = new HashMap<>();
+        statisticsMap.put(EVENT_STAT, toMap(statistics));
+        eventSink.success(statisticsMap);
+    }
+
+    public static int levelToInt(final Level level) {
+        return (level == null) ? Level.AV_LOG_TRACE.getValue() : level.getValue();
+    }
+
+    public static HashMap<String, String> toStringMap(final String key, final String value) {
+        final HashMap<String, String> map = new HashMap<>();
+        map.put(key, value);
+        return map;
+    }
+
+    public static HashMap<String, Integer> toIntMap(final String key, final int value) {
+        final HashMap<String, Integer> map = new HashMap<>();
+        map.put(key, value);
+        return map;
+    }
+
+    public static Map<String, Object> toMap(final Statistics statistics) {
+        final HashMap<String, Object> statisticsMap = new HashMap<>();
+
+        if (statistics != null) {
+            statisticsMap.put(KEY_STAT_TIME, statistics.getTime());
+            statisticsMap.put(KEY_STAT_SIZE, (statistics.getSize() < Integer.MAX_VALUE) ? (int) statistics.getSize() : (int) (statistics.getSize() % Integer.MAX_VALUE));
+            statisticsMap.put(KEY_STAT_BITRATE, statistics.getBitrate());
+            statisticsMap.put(KEY_STAT_SPEED, statistics.getSpeed());
+
+            statisticsMap.put(KEY_STAT_VIDEO_FRAME_NUMBER, statistics.getVideoFrameNumber());
+            statisticsMap.put(KEY_STAT_VIDEO_QUALITY, statistics.getVideoQuality());
+            statisticsMap.put(KEY_STAT_VIDEO_FPS, statistics.getVideoFps());
+        }
+
+        return statisticsMap;
+    }
+
+    public static HashMap<String, Object> toMediaInformationMap(final MediaInformation mediaInformation) {
+        final HashMap<String, Object> map = new HashMap<>();
+
+        if (mediaInformation != null) {
+            if (mediaInformation.getFormat() != null) {
+                map.put("format", mediaInformation.getFormat());
+            }
+            if (mediaInformation.getPath() != null) {
+                map.put("path", mediaInformation.getPath());
+            }
+            if (mediaInformation.getStartTime() != null) {
+                map.put("startTime", mediaInformation.getStartTime().intValue());
+            }
+            if (mediaInformation.getDuration() != null) {
+                map.put("duration", mediaInformation.getDuration().intValue());
+            }
+            if (mediaInformation.getBitrate() != null) {
+                map.put("bitrate", mediaInformation.getBitrate().intValue());
+            }
+            if (mediaInformation.getRawInformation() != null) {
+                map.put("rawInformation", mediaInformation.getRawInformation());
+            }
+
+            final Set<Map.Entry<String, String>> metadata = mediaInformation.getMetadataEntries();
+            if ((metadata != null) && (metadata.size() > 0)) {
+                final HashMap<String, String> metadataMap = new HashMap<>();
+
+                for (Map.Entry<String, String> entry : metadata) {
+                    metadataMap.put(entry.getKey(), entry.getValue());
+                }
+
+                map.put("metadata", metadataMap);
+            }
+
+            final List<StreamInformation> streams = mediaInformation.getStreams();
+            if ((streams != null) && (streams.size() > 0)) {
+                final ArrayList<Map<String, Object>> array = new ArrayList<>();
+
+                for (StreamInformation streamInformation : streams) {
+                    array.add(toStreamInformationMap(streamInformation));
+                }
+
+                map.put("streams", array);
+            }
+        }
+
+        return map;
+    }
+
+    public static Map<String, Object> toStreamInformationMap(final StreamInformation streamInformation) {
+        final HashMap<String, Object> map = new HashMap<>();
+
+        if (streamInformation != null) {
+            if (streamInformation.getIndex() != null) {
+                map.put("index", streamInformation.getIndex().intValue());
+            }
+            if (streamInformation.getType() != null) {
+                map.put("type", streamInformation.getType());
+            }
+            if (streamInformation.getCodec() != null) {
+                map.put("codec", streamInformation.getCodec());
+            }
+            if (streamInformation.getFullCodec() != null) {
+                map.put("fullCodec", streamInformation.getFullCodec());
+            }
+            if (streamInformation.getFormat() != null) {
+                map.put("format", streamInformation.getFormat());
+            }
+            if (streamInformation.getFullFormat() != null) {
+                map.put("fullFormat", streamInformation.getFullFormat());
+            }
+            if (streamInformation.getWidth() != null) {
+                map.put("width", streamInformation.getWidth().intValue());
+            }
+            if (streamInformation.getHeight() != null) {
+                map.put("height", streamInformation.getHeight().intValue());
+            }
+            if (streamInformation.getBitrate() != null) {
+                map.put("bitrate", streamInformation.getBitrate().intValue());
+            }
+            if (streamInformation.getSampleRate() != null) {
+                map.put("sampleRate", streamInformation.getSampleRate().intValue());
+            }
+            if (streamInformation.getSampleFormat() != null) {
+                map.put("sampleFormat", streamInformation.getSampleFormat());
+            }
+            if (streamInformation.getChannelLayout() != null) {
+                map.put("channelLayout", streamInformation.getChannelLayout());
+            }
+            if (streamInformation.getSampleAspectRatio() != null) {
+                map.put("sampleAspectRatio", streamInformation.getSampleAspectRatio());
+            }
+            if (streamInformation.getDisplayAspectRatio() != null) {
+                map.put("displayAspectRatio", streamInformation.getDisplayAspectRatio());
+            }
+            if (streamInformation.getAverageFrameRate() != null) {
+                map.put("averageFrameRate", streamInformation.getAverageFrameRate());
+            }
+            if (streamInformation.getRealFrameRate() != null) {
+                map.put("realFrameRate", streamInformation.getRealFrameRate());
+            }
+            if (streamInformation.getTimeBase() != null) {
+                map.put("timeBase", streamInformation.getTimeBase());
+            }
+            if (streamInformation.getCodecTimeBase() != null) {
+                map.put("codecTimeBase", streamInformation.getCodecTimeBase());
+            }
+
+            final Set<Map.Entry<String, String>> metadata = streamInformation.getMetadataEntries();
+            if ((metadata != null) && (metadata.size() > 0)) {
+                final HashMap<String, String> metadataMap = new HashMap<>();
+
+                for (Map.Entry<String, String> entry : metadata) {
+                    metadataMap.put(entry.getKey(), entry.getValue());
+                }
+
+                map.put("metadata", metadataMap);
+            }
+        }
+
+        return map;
+    }
+
+}

+ 36 - 0
packages/flutter_ffmpeg_https/ios/.gitignore

@@ -0,0 +1,36 @@
+.idea/
+.vagrant/
+.sconsign.dblite
+.svn/
+
+.DS_Store
+*.swp
+profile
+
+DerivedData/
+build/
+GeneratedPluginRegistrant.h
+GeneratedPluginRegistrant.m
+
+.generated/
+
+*.pbxuser
+*.mode1v3
+*.mode2v3
+*.perspectivev3
+
+!default.pbxuser
+!default.mode1v3
+!default.mode2v3
+!default.perspectivev3
+
+xcuserdata
+
+*.moved-aside
+
+*.pyc
+*sync/
+Icon?
+.tags*
+
+/Flutter/Generated.xcconfig

+ 0 - 0
packages/flutter_ffmpeg_https/ios/Assets/.gitkeep


+ 27 - 0
packages/flutter_ffmpeg_https/ios/Classes/FlutterFFmpegPlugin.h

@@ -0,0 +1,27 @@
+/*
+ * Copyright (c) 2019 Taner Sener
+ *
+ * This file is part of FlutterFFmpeg.
+ *
+ * FlutterFFmpeg is free software: you can redistribute it and/or modify
+ * it under the terms of the GNU Lesser General Public License as published by
+ * the Free Software Foundation, either version 3 of the License, or
+ * (at your option) any later version.
+ *
+ * FlutterFFmpeg is distributed in the hope that it will be useful,
+ * but WITHOUT ANY WARRANTY; without even the implied warranty of
+ * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
+ * GNU Lesser General Public License for more details.
+ *
+ * You should have received a copy of the GNU Lesser General Public License
+ * along with FlutterFFmpeg.  If not, see <http://www.gnu.org/licenses/>.
+ */
+
+#import <Flutter/Flutter.h>
+#import <mobileffmpeg/MobileFFmpegConfig.h>
+
+/**
+ * Flutter FFmpeg Plugin
+ */
+@interface FlutterFFmpegPlugin : NSObject<FlutterPlugin,FlutterStreamHandler,LogDelegate,StatisticsDelegate>
+@end

+ 387 - 0
packages/flutter_ffmpeg_https/ios/Classes/FlutterFfmpegPlugin.m

@@ -0,0 +1,387 @@
+/*
+ * Copyright (c) 2019 Taner Sener
+ *
+ * This file is part of FlutterFFmpeg.
+ *
+ * FlutterFFmpeg is free software: you can redistribute it and/or modify
+ * it under the terms of the GNU Lesser General Public License as published by
+ * the Free Software Foundation, either version 3 of the License, or
+ * (at your option) any later version.
+ *
+ * FlutterFFmpeg is distributed in the hope that it will be useful,
+ * but WITHOUT ANY WARRANTY; without even the implied warranty of
+ * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
+ * GNU Lesser General Public License for more details.
+ *
+ * You should have received a copy of the GNU Lesser General Public License
+ * along with FlutterFFmpeg.  If not, see <http://www.gnu.org/licenses/>.
+ */
+
+#import "FlutterFFmpegPlugin.h"
+
+#import <mobileffmpeg/ArchDetect.h>
+#import <mobileffmpeg/MobileFFmpeg.h>
+#import <mobileffmpeg/MobileFFmpegConfig.h>
+
+static NSString *const PLATFORM_NAME = @"ios";
+
+static NSString *const KEY_VERSION = @"version";
+static NSString *const KEY_RC = @"rc";
+static NSString *const KEY_PLATFORM = @"platform";
+static NSString *const KEY_PACKAGE_NAME = @"packageName";
+static NSString *const KEY_LAST_RC = @"lastRc";
+static NSString *const KEY_LAST_COMMAND_OUTPUT = @"lastCommandOutput";
+
+static NSString *const KEY_LOG_TEXT = @"log";
+static NSString *const KEY_LOG_LEVEL = @"level";
+
+static NSString *const KEY_STAT_TIME = @"time";
+static NSString *const KEY_STAT_SIZE = @"size";
+static NSString *const KEY_STAT_BITRATE = @"bitrate";
+static NSString *const KEY_STAT_SPEED = @"speed";
+static NSString *const KEY_STAT_VIDEO_FRAME_NUMBER = @"videoFrameNumber";
+static NSString *const KEY_STAT_VIDEO_QUALITY = @"videoQuality";
+static NSString *const KEY_STAT_VIDEO_FPS = @"videoFps";
+
+static NSString *const EVENT_LOG = @"FlutterFFmpegLogCallback";
+static NSString *const EVENT_STAT = @"FlutterFFmpegStatisticsCallback";
+
+/**
+ * Flutter FFmpeg Plugin
+ */
+@implementation FlutterFFmpegPlugin {
+    FlutterEventSink _eventSink;
+}
+
+- (FlutterError *)onListenWithArguments:(id)arguments eventSink:(FlutterEventSink)eventSink {
+    _eventSink = eventSink;
+    return nil;
+}
+
+- (FlutterError *)onCancelWithArguments:(id)arguments {
+    _eventSink = nil;
+    return nil;
+}
+
++ (void)registerWithRegistrar:(NSObject<FlutterPluginRegistrar>*)registrar {
+    FlutterFFmpegPlugin* instance = [[FlutterFFmpegPlugin alloc] init];
+
+    FlutterMethodChannel* methodChannel = [FlutterMethodChannel methodChannelWithName:@"flutter_ffmpeg" binaryMessenger:[registrar messenger]];
+    [registrar addMethodCallDelegate:instance channel:methodChannel];
+
+    FlutterEventChannel* eventChannel = [FlutterEventChannel eventChannelWithName:@"flutter_ffmpeg_event" binaryMessenger:[registrar messenger]];
+    [eventChannel setStreamHandler:instance];
+}
+
+- (void)handleMethodCall:(FlutterMethodCall*)call result:(FlutterResult)result {
+
+    // ARGUMENTS
+    NSArray* arguments = call.arguments[@"arguments"];
+    NSString* command = call.arguments[@"command"];
+    NSString* delimiter = call.arguments[@"delimiter"];
+
+    if ([@"getPlatform" isEqualToString:call.method]) {
+
+        NSString *architecture = [ArchDetect getArch];
+        result([FlutterFFmpegPlugin toStringDictionary:KEY_PLATFORM :[NSString stringWithFormat:@"%@-%@", PLATFORM_NAME, architecture]]);
+
+    } else if ([@"getFFmpegVersion" isEqualToString:call.method]) {
+
+        result([FlutterFFmpegPlugin toStringDictionary:KEY_VERSION :[MobileFFmpeg getFFmpegVersion]]);
+
+    } else if ([@"executeWithArguments" isEqualToString:call.method]) {
+
+        NSLog(@"Running FFmpeg with arguments: %@.\n", arguments);
+
+        dispatch_async(dispatch_get_global_queue(DISPATCH_QUEUE_PRIORITY_DEFAULT, 0), ^{
+
+            int rc = [MobileFFmpeg executeWithArguments:arguments];
+
+            NSLog(@"FFmpeg exited with rc: %d\n", rc);
+
+            result([FlutterFFmpegPlugin toIntDictionary:KEY_RC :[NSNumber numberWithInt:rc]]);
+        });
+
+    } else if ([@"execute" isEqualToString:call.method]) {
+
+        if (delimiter == nil) {
+            delimiter = @" ";
+        }
+
+        NSLog(@"Running FFmpeg command: %@ with delimiter %@.\n", command, delimiter);
+
+        dispatch_async(dispatch_get_global_queue(DISPATCH_QUEUE_PRIORITY_DEFAULT, 0), ^{
+            int rc = [MobileFFmpeg execute:command delimiter:delimiter];
+
+            NSLog(@"FFmpeg exited with rc: %d\n", rc);
+
+            result([FlutterFFmpegPlugin toIntDictionary:KEY_RC :[NSNumber numberWithInt:rc]]);
+        });
+
+    } else if ([@"cancel" isEqualToString:call.method]) {
+
+        [MobileFFmpeg cancel];
+
+    } else if ([@"enableRedirection" isEqualToString:call.method]) {
+
+        [MobileFFmpegConfig enableRedirection];
+
+    } else if ([@"disableRedirection" isEqualToString:call.method]) {
+
+        [MobileFFmpegConfig disableRedirection];
+
+    } else if ([@"getLogLevel" isEqualToString:call.method]) {
+
+        int logLevel = [MobileFFmpegConfig getLogLevel];
+        result([FlutterFFmpegPlugin toIntDictionary:KEY_LOG_LEVEL :[NSNumber numberWithInt:logLevel]]);
+
+    } else if ([@"setLogLevel" isEqualToString:call.method]) {
+
+        NSNumber* logLevel = call.arguments[@"level"];
+        [MobileFFmpegConfig setLogLevel:[logLevel intValue]];
+
+    } else if ([@"enableLogs" isEqualToString:call.method]) {
+
+        [MobileFFmpegConfig setLogDelegate:self];
+
+    } else if ([@"disableLogs" isEqualToString:call.method]) {
+
+        [MobileFFmpegConfig setLogDelegate:nil];
+
+    } else if ([@"enableStatistics" isEqualToString:call.method]) {
+
+        [MobileFFmpegConfig setStatisticsDelegate:self];
+
+    } else if ([@"disableStatistics" isEqualToString:call.method]) {
+
+        [MobileFFmpegConfig setStatisticsDelegate:nil];
+
+    } else if ([@"getLastReceivedStatistics" isEqualToString:call.method]) {
+
+        Statistics *statistics = [MobileFFmpegConfig getLastReceivedStatistics];
+        result([FlutterFFmpegPlugin toStatisticsDictionary:statistics]);
+
+    } else if ([@"resetStatistics" isEqualToString:call.method]) {
+
+        [MobileFFmpegConfig resetStatistics];
+
+    } else if ([@"setFontconfigConfigurationPath" isEqualToString:call.method]) {
+
+        NSString* path = call.arguments[@"path"];
+        [MobileFFmpegConfig setFontconfigConfigurationPath:path];
+
+    } else if ([@"setFontDirectory" isEqualToString:call.method]) {
+
+        NSString* fontDirectoryPath = call.arguments[@"fontDirectory"];
+        NSDictionary* fontNameMapping = call.arguments[@"fontNameMap"];
+        [MobileFFmpegConfig setFontDirectory:fontDirectoryPath with:fontNameMapping];
+
+    } else if ([@"getPackageName" isEqualToString:call.method]) {
+
+        NSString *packageName = [MobileFFmpegConfig getPackageName];
+        result([FlutterFFmpegPlugin toStringDictionary:KEY_PACKAGE_NAME :packageName]);
+
+    } else if ([@"getExternalLibraries" isEqualToString:call.method]) {
+
+        NSArray *externalLibraries = [MobileFFmpegConfig getExternalLibraries];
+        result(externalLibraries);
+
+    } else if ([@"getLastReturnCode" isEqualToString:call.method]) {
+
+        int lastReturnCode = [MobileFFmpeg getLastReturnCode];
+        result([FlutterFFmpegPlugin toIntDictionary:KEY_LAST_RC :[NSNumber numberWithInt:lastReturnCode]]);
+
+    } else if ([@"getLastCommandOutput" isEqualToString:call.method]) {
+
+        NSString *lastCommandOutput = [MobileFFmpeg getLastCommandOutput];
+        result([FlutterFFmpegPlugin toStringDictionary:KEY_LAST_COMMAND_OUTPUT :lastCommandOutput]);
+
+    } else if ([@"getMediaInformation" isEqualToString:call.method]) {
+
+        NSString* path = call.arguments[@"path"];
+        NSNumber* timeout = call.arguments[@"timeout"];
+
+        NSLog(@"Getting media information for %@ with timeout %d.\n", path, [timeout intValue]);
+
+        dispatch_async(dispatch_get_global_queue(DISPATCH_QUEUE_PRIORITY_DEFAULT, 0), ^{
+            MediaInformation *mediaInformation = [MobileFFmpeg getMediaInformation:path timeout:[timeout intValue]];
+            result([FlutterFFmpegPlugin toMediaInformationDictionary:mediaInformation]);
+        });
+
+    } else {
+
+        result(FlutterMethodNotImplemented);
+
+    }
+}
+
+- (void)logCallback: (int)level :(NSString*)message {
+    dispatch_async(dispatch_get_main_queue(), ^{
+        NSMutableDictionary *dictionary = [[NSMutableDictionary alloc] init];
+        dictionary[KEY_LOG_LEVEL] = [NSNumber numberWithInt:level];
+        dictionary[KEY_LOG_TEXT] = message;
+
+        [self emitLogMessage: dictionary];
+    });
+}
+
+- (void)statisticsCallback:(Statistics *)statistics {
+    dispatch_async(dispatch_get_main_queue(), ^{
+        [self emitStatistics: statistics];
+    });
+}
+
+- (void)emitLogMessage:(NSDictionary*)logMessage{
+    _eventSink([FlutterFFmpegPlugin toStringDictionary:EVENT_LOG :logMessage]);
+}
+
+- (void)emitStatistics:(Statistics*)statistics{
+    NSDictionary *dictionary = [FlutterFFmpegPlugin toStatisticsDictionary:statistics];
+    _eventSink([FlutterFFmpegPlugin toStringDictionary:EVENT_STAT :dictionary]);
+}
+
++ (NSDictionary *)toStringDictionary:(NSString*)key :(NSString*)value {
+    NSMutableDictionary *dictionary = [[NSMutableDictionary alloc] init];
+    dictionary[key] = value;
+
+    return dictionary;
+}
+
++ (NSDictionary *)toIntDictionary:(NSString*)key :(NSNumber*)value {
+    NSMutableDictionary *dictionary = [[NSMutableDictionary alloc] init];
+    dictionary[key] = value;
+
+    return dictionary;
+}
+
++ (NSDictionary *)toStatisticsDictionary:(Statistics*)statistics {
+    NSMutableDictionary *dictionary = [[NSMutableDictionary alloc] init];
+
+    if (statistics != nil) {
+        dictionary[KEY_STAT_TIME] = [NSNumber numberWithInt: [statistics getTime]];
+        dictionary[KEY_STAT_SIZE] = [NSNumber numberWithLong: [statistics getSize]];
+
+        dictionary[KEY_STAT_BITRATE] = [NSNumber numberWithDouble: [statistics getBitrate]];
+        dictionary[KEY_STAT_SPEED] = [NSNumber numberWithDouble: [statistics getSpeed]];
+
+        dictionary[KEY_STAT_VIDEO_FRAME_NUMBER] = [NSNumber numberWithInt: [statistics getVideoFrameNumber]];
+        dictionary[KEY_STAT_VIDEO_QUALITY] = [NSNumber numberWithFloat: [statistics getVideoQuality]];
+        dictionary[KEY_STAT_VIDEO_FPS] = [NSNumber numberWithFloat: [statistics getVideoFps]];
+    }
+
+    return dictionary;
+}
+
++ (NSDictionary *)toMediaInformationDictionary:(MediaInformation*)mediaInformation {
+    NSMutableDictionary *dictionary = [[NSMutableDictionary alloc] init];
+
+    if (mediaInformation != nil) {
+        if ([mediaInformation getFormat] != nil) {
+            dictionary[@"format"] =  [mediaInformation getFormat];
+        }
+        if ([mediaInformation getPath] != nil) {
+            dictionary[@"path"] = [mediaInformation getPath];
+        }
+        if ([mediaInformation getStartTime] != nil) {
+            dictionary[@"startTime"] = [mediaInformation getStartTime];
+        }
+        if ([mediaInformation getDuration] != nil) {
+            dictionary[@"duration"] = [mediaInformation getDuration];
+        }
+        if ([mediaInformation getBitrate] != nil) {
+            dictionary[@"bitrate"] = [mediaInformation getBitrate];
+        }
+        if ([mediaInformation getRawInformation] != nil) {
+            dictionary[@"rawInformation"] = [mediaInformation getRawInformation];
+        }
+
+        NSDictionary *metadataDictionary = [mediaInformation getMetadataEntries];
+        if (metadataDictionary != nil && ([metadataDictionary count] > 0)) {
+            dictionary[@"metadata"] = metadataDictionary;
+        }
+
+        NSArray *streams = [mediaInformation getStreams];
+        if (streams != nil && ([streams count] > 0)) {
+            NSMutableArray *array = [[NSMutableArray alloc] init];
+
+            for (int i=0; i < [streams count]; i++) {
+                StreamInformation *streamInformation= [streams objectAtIndex:i];
+                [array addObject: [FlutterFFmpegPlugin toStreamInformationDictionary:streamInformation]];
+            }
+
+            dictionary[@"streams"] = array;
+        }
+    }
+
+    return dictionary;
+}
+
++ (NSDictionary *)toStreamInformationDictionary:(StreamInformation*)streamInformation {
+    NSMutableDictionary *dictionary = [[NSMutableDictionary alloc] init];
+
+    if (streamInformation != nil) {
+        if ([streamInformation getIndex] != nil) {
+            dictionary[@"index"] = [streamInformation getIndex];
+        }
+        if ([streamInformation getType] != nil) {
+            dictionary[@"type"] = [streamInformation getType];
+        }
+        if ([streamInformation getCodec] != nil) {
+            dictionary[@"codec"] = [streamInformation getCodec];
+        }
+        if ([streamInformation getFullCodec] != nil) {
+            dictionary[@"fullCodec"] = [streamInformation getFullCodec];
+        }
+        if ([streamInformation getFormat] != nil) {
+            dictionary[@"format"] = [streamInformation getFormat];
+        }
+        if ([streamInformation getFullFormat] != nil) {
+            dictionary[@"fullFormat"] = [streamInformation getFullFormat];
+        }
+        if ([streamInformation getWidth] != nil) {
+            dictionary[@"width"] = [streamInformation getWidth];
+        }
+        if ([streamInformation getHeight] != nil) {
+            dictionary[@"height"] = [streamInformation getHeight];
+        }
+        if ([streamInformation getBitrate] != nil) {
+            dictionary[@"bitrate"] = [streamInformation getBitrate];
+        }
+        if ([streamInformation getSampleRate] != nil) {
+            dictionary[@"sampleRate"] = [streamInformation getSampleRate];
+        }
+        if ([streamInformation getSampleFormat] != nil) {
+            dictionary[@"sampleFormat"] = [streamInformation getSampleFormat];
+        }
+        if ([streamInformation getChannelLayout] != nil) {
+            dictionary[@"channelLayout"] = [streamInformation getChannelLayout];
+        }
+        if ([streamInformation getSampleAspectRatio] != nil) {
+            dictionary[@"sampleAspectRatio"] = [streamInformation getSampleAspectRatio];
+        }
+        if ([streamInformation getDisplayAspectRatio] != nil) {
+            dictionary[@"displayAspectRatio"] = [streamInformation getDisplayAspectRatio];
+        }
+        if ([streamInformation getAverageFrameRate] != nil) {
+            dictionary[@"averageFrameRate"] = [streamInformation getAverageFrameRate];
+        }
+        if ([streamInformation getRealFrameRate] != nil) {
+            dictionary[@"realFrameRate"] = [streamInformation getRealFrameRate];
+        }
+        if ([streamInformation getTimeBase] != nil) {
+            dictionary[@"timeBase"] = [streamInformation getTimeBase];
+        }
+        if ([streamInformation getCodecTimeBase] != nil) {
+            dictionary[@"codecTimeBase"] = [streamInformation getCodecTimeBase];
+        }
+
+        NSDictionary *metadataDictionary = [streamInformation getMetadataEntries];
+        if (metadataDictionary != nil && ([metadataDictionary count] > 0)) {
+            dictionary[@"metadata"] = metadataDictionary;
+        }
+    }
+
+    return dictionary;
+}
+
+@end

Niektóre pliki nie zostały wyświetlone z powodu dużej ilości zmienionych plików