Ver código fonte

apply plugin template

Taner Sener 7 anos atrás
pai
commit
21d5bef574
87 arquivos alterados com 4533 adições e 2 exclusões
  1. 9 0
      .gitignore
  2. 10 0
      .metadata
  3. 2 0
      CHANGELOG.md
  4. 674 0
      LICENSE.GPLv3
  5. 165 0
      LICENSE.LGPLv3
  6. 3 2
      README.md
  7. 8 0
      android/.gitignore
  8. 38 0
      android/build.gradle
  9. 1 0
      android/gradle.properties
  10. 1 0
      android/settings.gradle
  11. 3 0
      android/src/main/AndroidManifest.xml
  12. 60 0
      android/src/main/java/com/arthenica/flutter/ffmpeg/FlutterFFmpegExecuteAsyncArgumentsTask.java
  13. 66 0
      android/src/main/java/com/arthenica/flutter/ffmpeg/FlutterFFmpegExecuteAsyncCommandTask.java
  14. 64 0
      android/src/main/java/com/arthenica/flutter/ffmpeg/FlutterFFmpegGetMediaInformationAsyncTask.java
  15. 435 0
      android/src/main/java/com/arthenica/flutter/ffmpeg/FlutterFFmpegPlugin.java
  16. 71 0
      example/.gitignore
  17. 10 0
      example/.metadata
  18. 16 0
      example/README.md
  19. 60 0
      example/android/app/build.gradle
  20. 41 0
      example/android/app/src/main/AndroidManifest.xml
  21. 13 0
      example/android/app/src/main/java/com/arthenica/flutter/ffmpeg/flutterffmpegexample/MainActivity.java
  22. 12 0
      example/android/app/src/main/res/drawable/launch_background.xml
  23. BIN
      example/android/app/src/main/res/mipmap-hdpi/ic_launcher.png
  24. BIN
      example/android/app/src/main/res/mipmap-mdpi/ic_launcher.png
  25. BIN
      example/android/app/src/main/res/mipmap-xhdpi/ic_launcher.png
  26. BIN
      example/android/app/src/main/res/mipmap-xxhdpi/ic_launcher.png
  27. BIN
      example/android/app/src/main/res/mipmap-xxxhdpi/ic_launcher.png
  28. 8 0
      example/android/app/src/main/res/values/styles.xml
  29. 29 0
      example/android/build.gradle
  30. 1 0
      example/android/gradle.properties
  31. 6 0
      example/android/gradle/wrapper/gradle-wrapper.properties
  32. 15 0
      example/android/settings.gradle
  33. BIN
      example/assets/colosseum.jpg
  34. BIN
      example/assets/pyramid.jpg
  35. BIN
      example/assets/tajmahal.jpg
  36. 26 0
      example/ios/Flutter/AppFrameworkInfo.plist
  37. 2 0
      example/ios/Flutter/Debug.xcconfig
  38. 2 0
      example/ios/Flutter/Release.xcconfig
  39. 69 0
      example/ios/Podfile
  40. 582 0
      example/ios/Runner.xcodeproj/project.pbxproj
  41. 7 0
      example/ios/Runner.xcodeproj/project.xcworkspace/contents.xcworkspacedata
  42. 93 0
      example/ios/Runner.xcodeproj/xcshareddata/xcschemes/Runner.xcscheme
  43. 10 0
      example/ios/Runner.xcworkspace/contents.xcworkspacedata
  44. 8 0
      example/ios/Runner.xcworkspace/xcshareddata/IDEWorkspaceChecks.plist
  45. 8 0
      example/ios/Runner.xcworkspace/xcshareddata/WorkspaceSettings.xcsettings
  46. 6 0
      example/ios/Runner/AppDelegate.h
  47. 13 0
      example/ios/Runner/AppDelegate.m
  48. 122 0
      example/ios/Runner/Assets.xcassets/AppIcon.appiconset/Contents.json
  49. BIN
      example/ios/Runner/Assets.xcassets/AppIcon.appiconset/Icon-App-1024x1024@1x.png
  50. BIN
      example/ios/Runner/Assets.xcassets/AppIcon.appiconset/Icon-App-20x20@1x.png
  51. BIN
      example/ios/Runner/Assets.xcassets/AppIcon.appiconset/Icon-App-20x20@2x.png
  52. BIN
      example/ios/Runner/Assets.xcassets/AppIcon.appiconset/Icon-App-20x20@3x.png
  53. BIN
      example/ios/Runner/Assets.xcassets/AppIcon.appiconset/Icon-App-29x29@1x.png
  54. BIN
      example/ios/Runner/Assets.xcassets/AppIcon.appiconset/Icon-App-29x29@2x.png
  55. BIN
      example/ios/Runner/Assets.xcassets/AppIcon.appiconset/Icon-App-29x29@3x.png
  56. BIN
      example/ios/Runner/Assets.xcassets/AppIcon.appiconset/Icon-App-40x40@1x.png
  57. BIN
      example/ios/Runner/Assets.xcassets/AppIcon.appiconset/Icon-App-40x40@2x.png
  58. BIN
      example/ios/Runner/Assets.xcassets/AppIcon.appiconset/Icon-App-40x40@3x.png
  59. BIN
      example/ios/Runner/Assets.xcassets/AppIcon.appiconset/Icon-App-60x60@2x.png
  60. BIN
      example/ios/Runner/Assets.xcassets/AppIcon.appiconset/Icon-App-60x60@3x.png
  61. BIN
      example/ios/Runner/Assets.xcassets/AppIcon.appiconset/Icon-App-76x76@1x.png
  62. BIN
      example/ios/Runner/Assets.xcassets/AppIcon.appiconset/Icon-App-76x76@2x.png
  63. BIN
      example/ios/Runner/Assets.xcassets/AppIcon.appiconset/Icon-App-83.5x83.5@2x.png
  64. 23 0
      example/ios/Runner/Assets.xcassets/LaunchImage.imageset/Contents.json
  65. BIN
      example/ios/Runner/Assets.xcassets/LaunchImage.imageset/LaunchImage.png
  66. BIN
      example/ios/Runner/Assets.xcassets/LaunchImage.imageset/LaunchImage@2x.png
  67. BIN
      example/ios/Runner/Assets.xcassets/LaunchImage.imageset/LaunchImage@3x.png
  68. 5 0
      example/ios/Runner/Assets.xcassets/LaunchImage.imageset/README.md
  69. 37 0
      example/ios/Runner/Base.lproj/LaunchScreen.storyboard
  70. 26 0
      example/ios/Runner/Base.lproj/Main.storyboard
  71. 45 0
      example/ios/Runner/Info.plist
  72. 9 0
      example/ios/Runner/main.m
  73. 41 0
      example/lib/flutter_ffmpeg_test_app.dart
  74. 547 0
      example/lib/flutter_ffmpeg_test_app_state.dart
  75. 5 0
      example/lib/main.dart
  76. 46 0
      example/lib/video_util.dart
  77. 23 0
      example/pubspec.yaml
  78. 26 0
      example/test/widget_test.dart
  79. 20 0
      flutter_ffmpeg.iml
  80. 36 0
      ios/.gitignore
  81. 0 0
      ios/Assets/.gitkeep
  82. 24 0
      ios/Classes/FlutterFFmpegPlugin.h
  83. 384 0
      ios/Classes/FlutterFfmpegPlugin.m
  84. 22 0
      ios/flutter_ffmpeg.podspec
  85. 352 0
      lib/flutter_ffmpeg.dart
  86. 76 0
      lib/log_level.dart
  87. 17 0
      pubspec.yaml

+ 9 - 0
.gitignore

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

+ 10 - 0
.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

+ 2 - 0
CHANGELOG.md

@@ -0,0 +1,2 @@
+## 0.1.0
+- First release

+ 674 - 0
LICENSE.GPLv3

@@ -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>.

+ 165 - 0
LICENSE.LGPLv3

@@ -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.

+ 3 - 2
README.md

@@ -1,2 +1,3 @@
-# flutter-ffmpeg
-FFmpeg for flutter
+# flutter_ffmpeg
+
+FFmpeg plugin for Flutter. Supports iOS and Android.

+ 8 - 0
android/.gitignore

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

+ 38 - 0
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
android/gradle.properties

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

+ 1 - 0
android/settings.gradle

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

+ 3 - 0
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>

+ 60 - 0
android/src/main/java/com/arthenica/flutter/ffmpeg/FlutterFFmpegExecuteAsyncArgumentsTask.java

@@ -0,0 +1,60 @@
+/*
+ * 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;
+
+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));
+    }
+
+}

+ 66 - 0
android/src/main/java/com/arthenica/flutter/ffmpeg/FlutterFFmpegExecuteAsyncCommandTask.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 io.flutter.plugin.common.MethodChannel;
+
+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));
+    }
+
+}

+ 64 - 0
android/src/main/java/com/arthenica/flutter/ffmpeg/FlutterFFmpegGetMediaInformationAsyncTask.java

@@ -0,0 +1,64 @@
+/*
+ * 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;
+
+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));
+    }
+
+}

+ 435 - 0
android/src/main/java/com/arthenica/flutter/ffmpeg/FlutterFFmpegPlugin.java

@@ -0,0 +1,435 @@
+/*
+ * 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;
+
+/**
+ * <p>Main class of Flutter FFmpeg Plugin.
+ *
+ * @author Taner Sener
+ */
+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;
+    }
+
+}

+ 71 - 0
example/.gitignore

@@ -0,0 +1,71 @@
+# Miscellaneous
+*.class
+*.lock
+*.log
+*.pyc
+*.swp
+.DS_Store
+.atom/
+.buildlog/
+.history
+.svn/
+
+# IntelliJ related
+*.iml
+*.ipr
+*.iws
+.idea/
+
+# Visual Studio Code related
+.vscode/
+
+# Flutter/Dart/Pub related
+**/doc/api/
+.dart_tool/
+.flutter-plugins
+.packages
+.pub-cache/
+.pub/
+build/
+
+# Android related
+**/android/**/gradle-wrapper.jar
+**/android/.gradle
+**/android/captures/
+**/android/gradlew
+**/android/gradlew.bat
+**/android/local.properties
+**/android/**/GeneratedPluginRegistrant.java
+
+# iOS/XCode related
+**/ios/**/*.mode1v3
+**/ios/**/*.mode2v3
+**/ios/**/*.moved-aside
+**/ios/**/*.pbxuser
+**/ios/**/*.perspectivev3
+**/ios/**/*sync/
+**/ios/**/.sconsign.dblite
+**/ios/**/.tags*
+**/ios/**/.vagrant/
+**/ios/**/DerivedData/
+**/ios/**/Icon?
+**/ios/**/Pods/
+**/ios/**/.symlinks/
+**/ios/**/profile
+**/ios/**/xcuserdata
+**/ios/.generated/
+**/ios/Flutter/App.framework
+**/ios/Flutter/Flutter.framework
+**/ios/Flutter/Generated.xcconfig
+**/ios/Flutter/app.flx
+**/ios/Flutter/app.zip
+**/ios/Flutter/flutter_assets/
+**/ios/ServiceDefinitions.json
+**/ios/Runner/GeneratedPluginRegistrant.*
+
+# Exceptions to above rules.
+!**/ios/**/default.mode1v3
+!**/ios/**/default.mode2v3
+!**/ios/**/default.pbxuser
+!**/ios/**/default.perspectivev3
+!/packages/flutter_tools/test/data/dart_dependencies_test/**/.packages

+ 10 - 0
example/.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: app

+ 16 - 0
example/README.md

@@ -0,0 +1,16 @@
+# flutter_ffmpeg_example
+
+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.

+ 60 - 0
example/android/app/build.gradle

@@ -0,0 +1,60 @@
+def localProperties = new Properties()
+def localPropertiesFile = rootProject.file('local.properties')
+if (localPropertiesFile.exists()) {
+    localPropertiesFile.withReader('UTF-8') { reader ->
+        localProperties.load(reader)
+    }
+}
+
+def flutterRoot = localProperties.getProperty('flutter.sdk')
+if (flutterRoot == null) {
+    throw new GradleException("Flutter SDK not found. Define location with flutter.sdk in the local.properties file.")
+}
+
+def flutterVersionCode = localProperties.getProperty('flutter.versionCode')
+if (flutterVersionCode == null) {
+    flutterVersionCode = '1'
+}
+
+def flutterVersionName = localProperties.getProperty('flutter.versionName')
+if (flutterVersionName == null) {
+    flutterVersionName = '1.0'
+}
+
+apply plugin: 'com.android.application'
+apply from: "$flutterRoot/packages/flutter_tools/gradle/flutter.gradle"
+
+android {
+    compileSdkVersion 27
+
+    lintOptions {
+        disable 'InvalidPackage'
+    }
+
+    defaultConfig {
+        applicationId "com.arthenica.flutter.ffmpeg.FlutterFFmpegExample"
+        minSdkVersion 21
+        targetSdkVersion 27
+        versionCode flutterVersionCode.toInteger()
+        versionName flutterVersionName
+        testInstrumentationRunner "android.support.test.runner.AndroidJUnitRunner"
+    }
+
+    buildTypes {
+        release {
+            // TODO: Add your own signing config for the release build.
+            // Signing with the debug keys for now, so `flutter run --release` works.
+            signingConfig signingConfigs.debug
+        }
+    }
+}
+
+flutter {
+    source '../..'
+}
+
+dependencies {
+    testImplementation 'junit:junit:4.12'
+    androidTestImplementation 'com.android.support.test:runner:1.0.2'
+    androidTestImplementation 'com.android.support.test.espresso:espresso-core:3.0.2'
+}

+ 41 - 0
example/android/app/src/main/AndroidManifest.xml

@@ -0,0 +1,41 @@
+<manifest xmlns:android="http://schemas.android.com/apk/res/android"
+    xmlns:tools="http://schemas.android.com/tools"
+    package="com.arthenica.flutter.ffmpeg.flutterffmpegexample">
+
+    <!-- The INTERNET permission is required for development. Specifically,
+         flutter needs it to communicate with the running application
+         to allow setting breakpoints, to provide hot reload, etc.
+    -->
+    <uses-permission android:name="android.permission.INTERNET"/>
+
+    <!-- io.flutter.app.FlutterApplication is an android.app.Application that
+         calls FlutterMain.startInitialization(this); in its onCreate method.
+         In most cases you can leave this as-is, but you if you want to provide
+         additional functionality it is fine to subclass or reimplement
+         FlutterApplication and put your custom class here. -->
+    <application
+        android:name="io.flutter.app.FlutterApplication"
+        android:label="flutter_ffmpeg_example"
+        android:icon="@mipmap/ic_launcher"
+        tools:replace="android:label">
+        <activity
+            android:name=".MainActivity"
+            android:launchMode="singleTop"
+            android:theme="@style/LaunchTheme"
+            android:configChanges="orientation|keyboardHidden|keyboard|screenSize|locale|layoutDirection|fontScale|screenLayout|density"
+            android:hardwareAccelerated="true"
+            android:windowSoftInputMode="adjustResize">
+            <!-- This keeps the window background of the activity showing
+                 until Flutter renders its first frame. It can be removed if
+                 there is no splash screen (such as the default splash screen
+                 defined in @style/LaunchTheme). -->
+            <meta-data
+                android:name="io.flutter.app.android.SplashScreenUntilFirstFrame"
+                android:value="true" />
+            <intent-filter>
+                <action android:name="android.intent.action.MAIN"/>
+                <category android:name="android.intent.category.LAUNCHER"/>
+            </intent-filter>
+        </activity>
+    </application>
+</manifest>

+ 13 - 0
example/android/app/src/main/java/com/arthenica/flutter/ffmpeg/flutterffmpegexample/MainActivity.java

@@ -0,0 +1,13 @@
+package com.arthenica.flutter.ffmpeg.flutterffmpegexample;
+
+import android.os.Bundle;
+import io.flutter.app.FlutterActivity;
+import io.flutter.plugins.GeneratedPluginRegistrant;
+
+public class MainActivity extends FlutterActivity {
+  @Override
+  protected void onCreate(Bundle savedInstanceState) {
+    super.onCreate(savedInstanceState);
+    GeneratedPluginRegistrant.registerWith(this);
+  }
+}

+ 12 - 0
example/android/app/src/main/res/drawable/launch_background.xml

@@ -0,0 +1,12 @@
+<?xml version="1.0" encoding="utf-8"?>
+<!-- Modify this file to customize your launch splash screen -->
+<layer-list xmlns:android="http://schemas.android.com/apk/res/android">
+    <item android:drawable="@android:color/white" />
+
+    <!-- You can insert your own image assets here -->
+    <!-- <item>
+        <bitmap
+            android:gravity="center"
+            android:src="@mipmap/launch_image" />
+    </item> -->
+</layer-list>

BIN
example/android/app/src/main/res/mipmap-hdpi/ic_launcher.png


BIN
example/android/app/src/main/res/mipmap-mdpi/ic_launcher.png


BIN
example/android/app/src/main/res/mipmap-xhdpi/ic_launcher.png


BIN
example/android/app/src/main/res/mipmap-xxhdpi/ic_launcher.png


BIN
example/android/app/src/main/res/mipmap-xxxhdpi/ic_launcher.png


+ 8 - 0
example/android/app/src/main/res/values/styles.xml

@@ -0,0 +1,8 @@
+<?xml version="1.0" encoding="utf-8"?>
+<resources>
+    <style name="LaunchTheme" parent="@android:style/Theme.Black.NoTitleBar">
+        <!-- Show a splash screen on the activity. Automatically removed when
+             Flutter draws its first frame -->
+        <item name="android:windowBackground">@drawable/launch_background</item>
+    </style>
+</resources>

+ 29 - 0
example/android/build.gradle

@@ -0,0 +1,29 @@
+buildscript {
+    repositories {
+        google()
+        jcenter()
+    }
+
+    dependencies {
+        classpath 'com.android.tools.build:gradle:3.2.1'
+    }
+}
+
+allprojects {
+    repositories {
+        google()
+        jcenter()
+    }
+}
+
+rootProject.buildDir = '../build'
+subprojects {
+    project.buildDir = "${rootProject.buildDir}/${project.name}"
+}
+subprojects {
+    project.evaluationDependsOn(':app')
+}
+
+task clean(type: Delete) {
+    delete rootProject.buildDir
+}

+ 1 - 0
example/android/gradle.properties

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

+ 6 - 0
example/android/gradle/wrapper/gradle-wrapper.properties

@@ -0,0 +1,6 @@
+#Fri Jun 23 08:50:38 CEST 2017
+distributionBase=GRADLE_USER_HOME
+distributionPath=wrapper/dists
+zipStoreBase=GRADLE_USER_HOME
+zipStorePath=wrapper/dists
+distributionUrl=https\://services.gradle.org/distributions/gradle-4.10.2-all.zip

+ 15 - 0
example/android/settings.gradle

@@ -0,0 +1,15 @@
+include ':app'
+
+def flutterProjectRoot = rootProject.projectDir.parentFile.toPath()
+
+def plugins = new Properties()
+def pluginsFile = new File(flutterProjectRoot.toFile(), '.flutter-plugins')
+if (pluginsFile.exists()) {
+    pluginsFile.withReader('UTF-8') { reader -> plugins.load(reader) }
+}
+
+plugins.each { name, path ->
+    def pluginDirectory = flutterProjectRoot.resolve(path).resolve('android').toFile()
+    include ":$name"
+    project(":$name").projectDir = pluginDirectory
+}

BIN
example/assets/colosseum.jpg


BIN
example/assets/pyramid.jpg


BIN
example/assets/tajmahal.jpg


+ 26 - 0
example/ios/Flutter/AppFrameworkInfo.plist

@@ -0,0 +1,26 @@
+<?xml version="1.0" encoding="UTF-8"?>
+<!DOCTYPE plist PUBLIC "-//Apple//DTD PLIST 1.0//EN" "http://www.apple.com/DTDs/PropertyList-1.0.dtd">
+<plist version="1.0">
+<dict>
+  <key>CFBundleDevelopmentRegion</key>
+  <string>en</string>
+  <key>CFBundleExecutable</key>
+  <string>App</string>
+  <key>CFBundleIdentifier</key>
+  <string>io.flutter.flutter.app</string>
+  <key>CFBundleInfoDictionaryVersion</key>
+  <string>6.0</string>
+  <key>CFBundleName</key>
+  <string>App</string>
+  <key>CFBundlePackageType</key>
+  <string>FMWK</string>
+  <key>CFBundleShortVersionString</key>
+  <string>1.0</string>
+  <key>CFBundleSignature</key>
+  <string>????</string>
+  <key>CFBundleVersion</key>
+  <string>1.0</string>
+  <key>MinimumOSVersion</key>
+  <string>8.0</string>
+</dict>
+</plist>

+ 2 - 0
example/ios/Flutter/Debug.xcconfig

@@ -0,0 +1,2 @@
+#include "Pods/Target Support Files/Pods-Runner/Pods-Runner.debug.xcconfig"
+#include "Generated.xcconfig"

+ 2 - 0
example/ios/Flutter/Release.xcconfig

@@ -0,0 +1,2 @@
+#include "Pods/Target Support Files/Pods-Runner/Pods-Runner.release.xcconfig"
+#include "Generated.xcconfig"

+ 69 - 0
example/ios/Podfile

@@ -0,0 +1,69 @@
+# Uncomment this line to define a global platform for your project
+platform :ios, '9.3'
+
+# CocoaPods analytics sends network stats synchronously affecting flutter build latency.
+ENV['COCOAPODS_DISABLE_STATS'] = 'true'
+
+project 'Runner', {
+  'Debug' => :debug,
+  'Profile' => :release,
+  'Release' => :release,
+}
+
+def parse_KV_file(file, separator='=')
+  file_abs_path = File.expand_path(file)
+  if !File.exists? file_abs_path
+    return [];
+  end
+  pods_ary = []
+  skip_line_start_symbols = ["#", "/"]
+  File.foreach(file_abs_path) { |line|
+      next if skip_line_start_symbols.any? { |symbol| line =~ /^\s*#{symbol}/ }
+      plugin = line.split(pattern=separator)
+      if plugin.length == 2
+        podname = plugin[0].strip()
+        path = plugin[1].strip()
+        podpath = File.expand_path("#{path}", file_abs_path)
+        pods_ary.push({:name => podname, :path => podpath});
+      else
+        puts "Invalid plugin specification: #{line}"
+      end
+  }
+  return pods_ary
+end
+
+target 'Runner' do
+  # Prepare symlinks folder. We use symlinks to avoid having Podfile.lock
+  # referring to absolute paths on developers' machines.
+  system('rm -rf .symlinks')
+  system('mkdir -p .symlinks/plugins')
+
+  # Flutter Pods
+  generated_xcode_build_settings = parse_KV_file('./Flutter/Generated.xcconfig')
+  if generated_xcode_build_settings.empty?
+    puts "Generated.xcconfig must exist. If you're running pod install manually, make sure flutter packages get is executed first."
+  end
+  generated_xcode_build_settings.map { |p|
+    if p[:name] == 'FLUTTER_FRAMEWORK_DIR'
+      symlink = File.join('.symlinks', 'flutter')
+      File.symlink(File.dirname(p[:path]), symlink)
+      pod 'Flutter', :path => File.join(symlink, File.basename(p[:path]))
+    end
+  }
+
+  # Plugin Pods
+  plugin_pods = parse_KV_file('../.flutter-plugins')
+  plugin_pods.map { |p|
+    symlink = File.join('.symlinks', 'plugins', p[:name])
+    File.symlink(p[:path], symlink)
+    pod p[:name], :path => File.join(symlink, 'ios')
+  }
+end
+
+post_install do |installer|
+  installer.pods_project.targets.each do |target|
+    target.build_configurations.each do |config|
+      config.build_settings['ENABLE_BITCODE'] = 'NO'
+    end
+  end
+end

+ 582 - 0
example/ios/Runner.xcodeproj/project.pbxproj

@@ -0,0 +1,582 @@
+// !$*UTF8*$!
+{
+	archiveVersion = 1;
+	classes = {
+	};
+	objectVersion = 46;
+	objects = {
+
+/* Begin PBXBuildFile section */
+		1498D2341E8E89220040F4C2 /* GeneratedPluginRegistrant.m in Sources */ = {isa = PBXBuildFile; fileRef = 1498D2331E8E89220040F4C2 /* GeneratedPluginRegistrant.m */; };
+		2D5378261FAA1A9400D5DBA9 /* flutter_assets in Resources */ = {isa = PBXBuildFile; fileRef = 2D5378251FAA1A9400D5DBA9 /* flutter_assets */; };
+		3B3967161E833CAA004F5970 /* AppFrameworkInfo.plist in Resources */ = {isa = PBXBuildFile; fileRef = 3B3967151E833CAA004F5970 /* AppFrameworkInfo.plist */; };
+		3B80C3941E831B6300D905FE /* App.framework in Frameworks */ = {isa = PBXBuildFile; fileRef = 3B80C3931E831B6300D905FE /* App.framework */; };
+		3B80C3951E831B6300D905FE /* App.framework in Embed Frameworks */ = {isa = PBXBuildFile; fileRef = 3B80C3931E831B6300D905FE /* App.framework */; settings = {ATTRIBUTES = (CodeSignOnCopy, RemoveHeadersOnCopy, ); }; };
+		9705A1C61CF904A100538489 /* Flutter.framework in Frameworks */ = {isa = PBXBuildFile; fileRef = 9740EEBA1CF902C7004384FC /* Flutter.framework */; };
+		9705A1C71CF904A300538489 /* Flutter.framework in Embed Frameworks */ = {isa = PBXBuildFile; fileRef = 9740EEBA1CF902C7004384FC /* Flutter.framework */; settings = {ATTRIBUTES = (CodeSignOnCopy, RemoveHeadersOnCopy, ); }; };
+		9740EEB41CF90195004384FC /* Debug.xcconfig in Resources */ = {isa = PBXBuildFile; fileRef = 9740EEB21CF90195004384FC /* Debug.xcconfig */; };
+		978B8F6F1D3862AE00F588F7 /* AppDelegate.m in Sources */ = {isa = PBXBuildFile; fileRef = 7AFFD8EE1D35381100E5BB4D /* AppDelegate.m */; };
+		97C146F31CF9000F007C117D /* main.m in Sources */ = {isa = PBXBuildFile; fileRef = 97C146F21CF9000F007C117D /* main.m */; };
+		97C146FC1CF9000F007C117D /* Main.storyboard in Resources */ = {isa = PBXBuildFile; fileRef = 97C146FA1CF9000F007C117D /* Main.storyboard */; };
+		97C146FE1CF9000F007C117D /* Assets.xcassets in Resources */ = {isa = PBXBuildFile; fileRef = 97C146FD1CF9000F007C117D /* Assets.xcassets */; };
+		97C147011CF9000F007C117D /* LaunchScreen.storyboard in Resources */ = {isa = PBXBuildFile; fileRef = 97C146FF1CF9000F007C117D /* LaunchScreen.storyboard */; };
+		B22DCA878D98E0611C94A440 /* libPods-Runner.a in Frameworks */ = {isa = PBXBuildFile; fileRef = 091F5A393B969ED7BA77DFEA /* libPods-Runner.a */; };
+/* End PBXBuildFile section */
+
+/* Begin PBXCopyFilesBuildPhase section */
+		9705A1C41CF9048500538489 /* Embed Frameworks */ = {
+			isa = PBXCopyFilesBuildPhase;
+			buildActionMask = 2147483647;
+			dstPath = "";
+			dstSubfolderSpec = 10;
+			files = (
+				3B80C3951E831B6300D905FE /* App.framework in Embed Frameworks */,
+				9705A1C71CF904A300538489 /* Flutter.framework in Embed Frameworks */,
+			);
+			name = "Embed Frameworks";
+			runOnlyForDeploymentPostprocessing = 0;
+		};
+/* End PBXCopyFilesBuildPhase section */
+
+/* Begin PBXFileReference section */
+		091F5A393B969ED7BA77DFEA /* libPods-Runner.a */ = {isa = PBXFileReference; explicitFileType = archive.ar; includeInIndex = 0; path = "libPods-Runner.a"; sourceTree = BUILT_PRODUCTS_DIR; };
+		1498D2321E8E86230040F4C2 /* GeneratedPluginRegistrant.h */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.c.h; path = GeneratedPluginRegistrant.h; sourceTree = "<group>"; };
+		1498D2331E8E89220040F4C2 /* GeneratedPluginRegistrant.m */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.c.objc; path = GeneratedPluginRegistrant.m; sourceTree = "<group>"; };
+		2D5378251FAA1A9400D5DBA9 /* flutter_assets */ = {isa = PBXFileReference; lastKnownFileType = folder; name = flutter_assets; path = Flutter/flutter_assets; sourceTree = SOURCE_ROOT; };
+		3B3967151E833CAA004F5970 /* AppFrameworkInfo.plist */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = text.plist.xml; name = AppFrameworkInfo.plist; path = Flutter/AppFrameworkInfo.plist; sourceTree = "<group>"; };
+		3B80C3931E831B6300D905FE /* App.framework */ = {isa = PBXFileReference; lastKnownFileType = wrapper.framework; name = App.framework; path = Flutter/App.framework; sourceTree = "<group>"; };
+		4B23FBE71044F4775C377690 /* Pods-Runner.release.xcconfig */ = {isa = PBXFileReference; includeInIndex = 1; lastKnownFileType = text.xcconfig; name = "Pods-Runner.release.xcconfig"; path = "Target Support Files/Pods-Runner/Pods-Runner.release.xcconfig"; sourceTree = "<group>"; };
+		57FE3D43E457162CAB644DD1 /* Pods-Runner.profile.xcconfig */ = {isa = PBXFileReference; includeInIndex = 1; lastKnownFileType = text.xcconfig; name = "Pods-Runner.profile.xcconfig"; path = "Target Support Files/Pods-Runner/Pods-Runner.profile.xcconfig"; sourceTree = "<group>"; };
+		691935F79955E2AC17A18B11 /* Pods-Runner.debug.xcconfig */ = {isa = PBXFileReference; includeInIndex = 1; lastKnownFileType = text.xcconfig; name = "Pods-Runner.debug.xcconfig"; path = "Target Support Files/Pods-Runner/Pods-Runner.debug.xcconfig"; sourceTree = "<group>"; };
+		7AFA3C8E1D35360C0083082E /* Release.xcconfig */ = {isa = PBXFileReference; lastKnownFileType = text.xcconfig; name = Release.xcconfig; path = Flutter/Release.xcconfig; sourceTree = "<group>"; };
+		7AFFD8ED1D35381100E5BB4D /* AppDelegate.h */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.c.h; path = AppDelegate.h; sourceTree = "<group>"; };
+		7AFFD8EE1D35381100E5BB4D /* AppDelegate.m */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.c.objc; path = AppDelegate.m; sourceTree = "<group>"; };
+		9740EEB21CF90195004384FC /* Debug.xcconfig */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = text.xcconfig; name = Debug.xcconfig; path = Flutter/Debug.xcconfig; sourceTree = "<group>"; };
+		9740EEB31CF90195004384FC /* Generated.xcconfig */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = text.xcconfig; name = Generated.xcconfig; path = Flutter/Generated.xcconfig; sourceTree = "<group>"; };
+		9740EEBA1CF902C7004384FC /* Flutter.framework */ = {isa = PBXFileReference; lastKnownFileType = wrapper.framework; name = Flutter.framework; path = Flutter/Flutter.framework; sourceTree = "<group>"; };
+		97C146EE1CF9000F007C117D /* Runner.app */ = {isa = PBXFileReference; explicitFileType = wrapper.application; includeInIndex = 0; path = Runner.app; sourceTree = BUILT_PRODUCTS_DIR; };
+		97C146F21CF9000F007C117D /* main.m */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.c.objc; path = main.m; sourceTree = "<group>"; };
+		97C146FB1CF9000F007C117D /* Base */ = {isa = PBXFileReference; lastKnownFileType = file.storyboard; name = Base; path = Base.lproj/Main.storyboard; sourceTree = "<group>"; };
+		97C146FD1CF9000F007C117D /* Assets.xcassets */ = {isa = PBXFileReference; lastKnownFileType = folder.assetcatalog; path = Assets.xcassets; sourceTree = "<group>"; };
+		97C147001CF9000F007C117D /* Base */ = {isa = PBXFileReference; lastKnownFileType = file.storyboard; name = Base; path = Base.lproj/LaunchScreen.storyboard; sourceTree = "<group>"; };
+		97C147021CF9000F007C117D /* Info.plist */ = {isa = PBXFileReference; lastKnownFileType = text.plist.xml; path = Info.plist; sourceTree = "<group>"; };
+/* End PBXFileReference section */
+
+/* Begin PBXFrameworksBuildPhase section */
+		97C146EB1CF9000F007C117D /* Frameworks */ = {
+			isa = PBXFrameworksBuildPhase;
+			buildActionMask = 2147483647;
+			files = (
+				9705A1C61CF904A100538489 /* Flutter.framework in Frameworks */,
+				3B80C3941E831B6300D905FE /* App.framework in Frameworks */,
+				B22DCA878D98E0611C94A440 /* libPods-Runner.a in Frameworks */,
+			);
+			runOnlyForDeploymentPostprocessing = 0;
+		};
+/* End PBXFrameworksBuildPhase section */
+
+/* Begin PBXGroup section */
+		9740EEB11CF90186004384FC /* Flutter */ = {
+			isa = PBXGroup;
+			children = (
+				2D5378251FAA1A9400D5DBA9 /* flutter_assets */,
+				3B80C3931E831B6300D905FE /* App.framework */,
+				3B3967151E833CAA004F5970 /* AppFrameworkInfo.plist */,
+				9740EEBA1CF902C7004384FC /* Flutter.framework */,
+				9740EEB21CF90195004384FC /* Debug.xcconfig */,
+				7AFA3C8E1D35360C0083082E /* Release.xcconfig */,
+				9740EEB31CF90195004384FC /* Generated.xcconfig */,
+			);
+			name = Flutter;
+			sourceTree = "<group>";
+		};
+		97C146E51CF9000F007C117D = {
+			isa = PBXGroup;
+			children = (
+				9740EEB11CF90186004384FC /* Flutter */,
+				97C146F01CF9000F007C117D /* Runner */,
+				97C146EF1CF9000F007C117D /* Products */,
+				D1255632BF1FA1BD23961730 /* Pods */,
+				C0AABD390DC764D327FE8EB6 /* Frameworks */,
+			);
+			sourceTree = "<group>";
+		};
+		97C146EF1CF9000F007C117D /* Products */ = {
+			isa = PBXGroup;
+			children = (
+				97C146EE1CF9000F007C117D /* Runner.app */,
+			);
+			name = Products;
+			sourceTree = "<group>";
+		};
+		97C146F01CF9000F007C117D /* Runner */ = {
+			isa = PBXGroup;
+			children = (
+				7AFFD8ED1D35381100E5BB4D /* AppDelegate.h */,
+				7AFFD8EE1D35381100E5BB4D /* AppDelegate.m */,
+				97C146FA1CF9000F007C117D /* Main.storyboard */,
+				97C146FD1CF9000F007C117D /* Assets.xcassets */,
+				97C146FF1CF9000F007C117D /* LaunchScreen.storyboard */,
+				97C147021CF9000F007C117D /* Info.plist */,
+				97C146F11CF9000F007C117D /* Supporting Files */,
+				1498D2321E8E86230040F4C2 /* GeneratedPluginRegistrant.h */,
+				1498D2331E8E89220040F4C2 /* GeneratedPluginRegistrant.m */,
+			);
+			path = Runner;
+			sourceTree = "<group>";
+		};
+		97C146F11CF9000F007C117D /* Supporting Files */ = {
+			isa = PBXGroup;
+			children = (
+				97C146F21CF9000F007C117D /* main.m */,
+			);
+			name = "Supporting Files";
+			sourceTree = "<group>";
+		};
+		C0AABD390DC764D327FE8EB6 /* Frameworks */ = {
+			isa = PBXGroup;
+			children = (
+				091F5A393B969ED7BA77DFEA /* libPods-Runner.a */,
+			);
+			name = Frameworks;
+			sourceTree = "<group>";
+		};
+		D1255632BF1FA1BD23961730 /* Pods */ = {
+			isa = PBXGroup;
+			children = (
+				691935F79955E2AC17A18B11 /* Pods-Runner.debug.xcconfig */,
+				4B23FBE71044F4775C377690 /* Pods-Runner.release.xcconfig */,
+				57FE3D43E457162CAB644DD1 /* Pods-Runner.profile.xcconfig */,
+			);
+			name = Pods;
+			path = Pods;
+			sourceTree = "<group>";
+		};
+/* End PBXGroup section */
+
+/* Begin PBXNativeTarget section */
+		97C146ED1CF9000F007C117D /* Runner */ = {
+			isa = PBXNativeTarget;
+			buildConfigurationList = 97C147051CF9000F007C117D /* Build configuration list for PBXNativeTarget "Runner" */;
+			buildPhases = (
+				21AF8DE8D9FF721314C951F3 /* [CP] Check Pods Manifest.lock */,
+				9740EEB61CF901F6004384FC /* Run Script */,
+				97C146EA1CF9000F007C117D /* Sources */,
+				97C146EB1CF9000F007C117D /* Frameworks */,
+				97C146EC1CF9000F007C117D /* Resources */,
+				9705A1C41CF9048500538489 /* Embed Frameworks */,
+				3B06AD1E1E4923F5004D2608 /* Thin Binary */,
+				E7D9C4F847D99ED1A9F12BBD /* [CP] Embed Pods Frameworks */,
+			);
+			buildRules = (
+			);
+			dependencies = (
+			);
+			name = Runner;
+			productName = Runner;
+			productReference = 97C146EE1CF9000F007C117D /* Runner.app */;
+			productType = "com.apple.product-type.application";
+		};
+/* End PBXNativeTarget section */
+
+/* Begin PBXProject section */
+		97C146E61CF9000F007C117D /* Project object */ = {
+			isa = PBXProject;
+			attributes = {
+				LastUpgradeCheck = 0910;
+				ORGANIZATIONNAME = "The Chromium Authors";
+				TargetAttributes = {
+					97C146ED1CF9000F007C117D = {
+						CreatedOnToolsVersion = 7.3.1;
+					};
+				};
+			};
+			buildConfigurationList = 97C146E91CF9000F007C117D /* Build configuration list for PBXProject "Runner" */;
+			compatibilityVersion = "Xcode 3.2";
+			developmentRegion = English;
+			hasScannedForEncodings = 0;
+			knownRegions = (
+				en,
+				Base,
+			);
+			mainGroup = 97C146E51CF9000F007C117D;
+			productRefGroup = 97C146EF1CF9000F007C117D /* Products */;
+			projectDirPath = "";
+			projectRoot = "";
+			targets = (
+				97C146ED1CF9000F007C117D /* Runner */,
+			);
+		};
+/* End PBXProject section */
+
+/* Begin PBXResourcesBuildPhase section */
+		97C146EC1CF9000F007C117D /* Resources */ = {
+			isa = PBXResourcesBuildPhase;
+			buildActionMask = 2147483647;
+			files = (
+				97C147011CF9000F007C117D /* LaunchScreen.storyboard in Resources */,
+				3B3967161E833CAA004F5970 /* AppFrameworkInfo.plist in Resources */,
+				9740EEB41CF90195004384FC /* Debug.xcconfig in Resources */,
+				97C146FE1CF9000F007C117D /* Assets.xcassets in Resources */,
+				2D5378261FAA1A9400D5DBA9 /* flutter_assets in Resources */,
+				97C146FC1CF9000F007C117D /* Main.storyboard in Resources */,
+			);
+			runOnlyForDeploymentPostprocessing = 0;
+		};
+/* End PBXResourcesBuildPhase section */
+
+/* Begin PBXShellScriptBuildPhase section */
+		21AF8DE8D9FF721314C951F3 /* [CP] Check Pods Manifest.lock */ = {
+			isa = PBXShellScriptBuildPhase;
+			buildActionMask = 2147483647;
+			files = (
+			);
+			inputFileListPaths = (
+			);
+			inputPaths = (
+				"${PODS_PODFILE_DIR_PATH}/Podfile.lock",
+				"${PODS_ROOT}/Manifest.lock",
+			);
+			name = "[CP] Check Pods Manifest.lock";
+			outputFileListPaths = (
+			);
+			outputPaths = (
+				"$(DERIVED_FILE_DIR)/Pods-Runner-checkManifestLockResult.txt",
+			);
+			runOnlyForDeploymentPostprocessing = 0;
+			shellPath = /bin/sh;
+			shellScript = "diff \"${PODS_PODFILE_DIR_PATH}/Podfile.lock\" \"${PODS_ROOT}/Manifest.lock\" > /dev/null\nif [ $? != 0 ] ; then\n    # print error to STDERR\n    echo \"error: The sandbox is not in sync with the Podfile.lock. Run 'pod install' or update your CocoaPods installation.\" >&2\n    exit 1\nfi\n# This output is used by Xcode 'outputs' to avoid re-running this script phase.\necho \"SUCCESS\" > \"${SCRIPT_OUTPUT_FILE_0}\"\n";
+			showEnvVarsInLog = 0;
+		};
+		3B06AD1E1E4923F5004D2608 /* Thin Binary */ = {
+			isa = PBXShellScriptBuildPhase;
+			buildActionMask = 2147483647;
+			files = (
+			);
+			inputPaths = (
+			);
+			name = "Thin Binary";
+			outputPaths = (
+			);
+			runOnlyForDeploymentPostprocessing = 0;
+			shellPath = /bin/sh;
+			shellScript = "/bin/sh \"$FLUTTER_ROOT/packages/flutter_tools/bin/xcode_backend.sh\" thin";
+		};
+		9740EEB61CF901F6004384FC /* Run Script */ = {
+			isa = PBXShellScriptBuildPhase;
+			buildActionMask = 2147483647;
+			files = (
+			);
+			inputPaths = (
+			);
+			name = "Run Script";
+			outputPaths = (
+			);
+			runOnlyForDeploymentPostprocessing = 0;
+			shellPath = /bin/sh;
+			shellScript = "/bin/sh \"$FLUTTER_ROOT/packages/flutter_tools/bin/xcode_backend.sh\" build";
+		};
+		E7D9C4F847D99ED1A9F12BBD /* [CP] Embed Pods Frameworks */ = {
+			isa = PBXShellScriptBuildPhase;
+			buildActionMask = 2147483647;
+			files = (
+			);
+			inputFileListPaths = (
+			);
+			inputPaths = (
+				"${PODS_ROOT}/Target Support Files/Pods-Runner/Pods-Runner-frameworks.sh",
+				"${PODS_ROOT}/../.symlinks/flutter/ios/Flutter.framework",
+			);
+			name = "[CP] Embed Pods Frameworks";
+			outputFileListPaths = (
+			);
+			outputPaths = (
+				"${TARGET_BUILD_DIR}/${FRAMEWORKS_FOLDER_PATH}/Flutter.framework",
+			);
+			runOnlyForDeploymentPostprocessing = 0;
+			shellPath = /bin/sh;
+			shellScript = "\"${PODS_ROOT}/Target Support Files/Pods-Runner/Pods-Runner-frameworks.sh\"\n";
+			showEnvVarsInLog = 0;
+		};
+/* End PBXShellScriptBuildPhase section */
+
+/* Begin PBXSourcesBuildPhase section */
+		97C146EA1CF9000F007C117D /* Sources */ = {
+			isa = PBXSourcesBuildPhase;
+			buildActionMask = 2147483647;
+			files = (
+				978B8F6F1D3862AE00F588F7 /* AppDelegate.m in Sources */,
+				97C146F31CF9000F007C117D /* main.m in Sources */,
+				1498D2341E8E89220040F4C2 /* GeneratedPluginRegistrant.m in Sources */,
+			);
+			runOnlyForDeploymentPostprocessing = 0;
+		};
+/* End PBXSourcesBuildPhase section */
+
+/* Begin PBXVariantGroup section */
+		97C146FA1CF9000F007C117D /* Main.storyboard */ = {
+			isa = PBXVariantGroup;
+			children = (
+				97C146FB1CF9000F007C117D /* Base */,
+			);
+			name = Main.storyboard;
+			sourceTree = "<group>";
+		};
+		97C146FF1CF9000F007C117D /* LaunchScreen.storyboard */ = {
+			isa = PBXVariantGroup;
+			children = (
+				97C147001CF9000F007C117D /* Base */,
+			);
+			name = LaunchScreen.storyboard;
+			sourceTree = "<group>";
+		};
+/* End PBXVariantGroup section */
+
+/* Begin XCBuildConfiguration section */
+		249021D3217E4FDB00AE95B9 /* Profile */ = {
+			isa = XCBuildConfiguration;
+			baseConfigurationReference = 7AFA3C8E1D35360C0083082E /* Release.xcconfig */;
+			buildSettings = {
+				ALWAYS_SEARCH_USER_PATHS = NO;
+				CLANG_ANALYZER_NONNULL = YES;
+				CLANG_CXX_LANGUAGE_STANDARD = "gnu++0x";
+				CLANG_CXX_LIBRARY = "libc++";
+				CLANG_ENABLE_MODULES = YES;
+				CLANG_ENABLE_OBJC_ARC = YES;
+				CLANG_WARN_BLOCK_CAPTURE_AUTORELEASING = YES;
+				CLANG_WARN_BOOL_CONVERSION = YES;
+				CLANG_WARN_COMMA = YES;
+				CLANG_WARN_CONSTANT_CONVERSION = YES;
+				CLANG_WARN_DIRECT_OBJC_ISA_USAGE = YES_ERROR;
+				CLANG_WARN_EMPTY_BODY = YES;
+				CLANG_WARN_ENUM_CONVERSION = YES;
+				CLANG_WARN_INFINITE_RECURSION = YES;
+				CLANG_WARN_INT_CONVERSION = YES;
+				CLANG_WARN_NON_LITERAL_NULL_CONVERSION = YES;
+				CLANG_WARN_OBJC_LITERAL_CONVERSION = YES;
+				CLANG_WARN_OBJC_ROOT_CLASS = YES_ERROR;
+				CLANG_WARN_RANGE_LOOP_ANALYSIS = YES;
+				CLANG_WARN_STRICT_PROTOTYPES = YES;
+				CLANG_WARN_SUSPICIOUS_MOVE = YES;
+				CLANG_WARN_UNREACHABLE_CODE = YES;
+				CLANG_WARN__DUPLICATE_METHOD_MATCH = YES;
+				"CODE_SIGN_IDENTITY[sdk=iphoneos*]" = "iPhone Developer";
+				COPY_PHASE_STRIP = NO;
+				DEBUG_INFORMATION_FORMAT = "dwarf-with-dsym";
+				ENABLE_NS_ASSERTIONS = NO;
+				ENABLE_STRICT_OBJC_MSGSEND = YES;
+				GCC_C_LANGUAGE_STANDARD = gnu99;
+				GCC_NO_COMMON_BLOCKS = YES;
+				GCC_WARN_64_TO_32_BIT_CONVERSION = YES;
+				GCC_WARN_ABOUT_RETURN_TYPE = YES_ERROR;
+				GCC_WARN_UNDECLARED_SELECTOR = YES;
+				GCC_WARN_UNINITIALIZED_AUTOS = YES_AGGRESSIVE;
+				GCC_WARN_UNUSED_FUNCTION = YES;
+				GCC_WARN_UNUSED_VARIABLE = YES;
+				IPHONEOS_DEPLOYMENT_TARGET = 8.0;
+				MTL_ENABLE_DEBUG_INFO = NO;
+				SDKROOT = iphoneos;
+				TARGETED_DEVICE_FAMILY = "1,2";
+				VALIDATE_PRODUCT = YES;
+			};
+			name = Profile;
+		};
+		249021D4217E4FDB00AE95B9 /* Profile */ = {
+			isa = XCBuildConfiguration;
+			baseConfigurationReference = 7AFA3C8E1D35360C0083082E /* Release.xcconfig */;
+			buildSettings = {
+				ASSETCATALOG_COMPILER_APPICON_NAME = AppIcon;
+				CURRENT_PROJECT_VERSION = "$(FLUTTER_BUILD_NUMBER)";
+				DEVELOPMENT_TEAM = S8QB4VV633;
+				ENABLE_BITCODE = NO;
+				FRAMEWORK_SEARCH_PATHS = (
+					"$(inherited)",
+					"$(PROJECT_DIR)/Flutter",
+				);
+				INFOPLIST_FILE = Runner/Info.plist;
+				LD_RUNPATH_SEARCH_PATHS = "$(inherited) @executable_path/Frameworks";
+				LIBRARY_SEARCH_PATHS = (
+					"$(inherited)",
+					"$(PROJECT_DIR)/Flutter",
+				);
+				PRODUCT_BUNDLE_IDENTIFIER = com.arthenica.flutter.ffmpeg.flutterFfmpegExample;
+				PRODUCT_NAME = "$(TARGET_NAME)";
+				VERSIONING_SYSTEM = "apple-generic";
+			};
+			name = Profile;
+		};
+		97C147031CF9000F007C117D /* Debug */ = {
+			isa = XCBuildConfiguration;
+			baseConfigurationReference = 9740EEB21CF90195004384FC /* Debug.xcconfig */;
+			buildSettings = {
+				ALWAYS_SEARCH_USER_PATHS = NO;
+				CLANG_ANALYZER_NONNULL = YES;
+				CLANG_CXX_LANGUAGE_STANDARD = "gnu++0x";
+				CLANG_CXX_LIBRARY = "libc++";
+				CLANG_ENABLE_MODULES = YES;
+				CLANG_ENABLE_OBJC_ARC = YES;
+				CLANG_WARN_BLOCK_CAPTURE_AUTORELEASING = YES;
+				CLANG_WARN_BOOL_CONVERSION = YES;
+				CLANG_WARN_COMMA = YES;
+				CLANG_WARN_CONSTANT_CONVERSION = YES;
+				CLANG_WARN_DIRECT_OBJC_ISA_USAGE = YES_ERROR;
+				CLANG_WARN_EMPTY_BODY = YES;
+				CLANG_WARN_ENUM_CONVERSION = YES;
+				CLANG_WARN_INFINITE_RECURSION = YES;
+				CLANG_WARN_INT_CONVERSION = YES;
+				CLANG_WARN_NON_LITERAL_NULL_CONVERSION = YES;
+				CLANG_WARN_OBJC_LITERAL_CONVERSION = YES;
+				CLANG_WARN_OBJC_ROOT_CLASS = YES_ERROR;
+				CLANG_WARN_RANGE_LOOP_ANALYSIS = YES;
+				CLANG_WARN_STRICT_PROTOTYPES = YES;
+				CLANG_WARN_SUSPICIOUS_MOVE = YES;
+				CLANG_WARN_UNREACHABLE_CODE = YES;
+				CLANG_WARN__DUPLICATE_METHOD_MATCH = YES;
+				"CODE_SIGN_IDENTITY[sdk=iphoneos*]" = "iPhone Developer";
+				COPY_PHASE_STRIP = NO;
+				DEBUG_INFORMATION_FORMAT = dwarf;
+				ENABLE_STRICT_OBJC_MSGSEND = YES;
+				ENABLE_TESTABILITY = YES;
+				GCC_C_LANGUAGE_STANDARD = gnu99;
+				GCC_DYNAMIC_NO_PIC = NO;
+				GCC_NO_COMMON_BLOCKS = YES;
+				GCC_OPTIMIZATION_LEVEL = 0;
+				GCC_PREPROCESSOR_DEFINITIONS = (
+					"DEBUG=1",
+					"$(inherited)",
+				);
+				GCC_WARN_64_TO_32_BIT_CONVERSION = YES;
+				GCC_WARN_ABOUT_RETURN_TYPE = YES_ERROR;
+				GCC_WARN_UNDECLARED_SELECTOR = YES;
+				GCC_WARN_UNINITIALIZED_AUTOS = YES_AGGRESSIVE;
+				GCC_WARN_UNUSED_FUNCTION = YES;
+				GCC_WARN_UNUSED_VARIABLE = YES;
+				IPHONEOS_DEPLOYMENT_TARGET = 8.0;
+				MTL_ENABLE_DEBUG_INFO = YES;
+				ONLY_ACTIVE_ARCH = YES;
+				SDKROOT = iphoneos;
+				TARGETED_DEVICE_FAMILY = "1,2";
+			};
+			name = Debug;
+		};
+		97C147041CF9000F007C117D /* Release */ = {
+			isa = XCBuildConfiguration;
+			baseConfigurationReference = 7AFA3C8E1D35360C0083082E /* Release.xcconfig */;
+			buildSettings = {
+				ALWAYS_SEARCH_USER_PATHS = NO;
+				CLANG_ANALYZER_NONNULL = YES;
+				CLANG_CXX_LANGUAGE_STANDARD = "gnu++0x";
+				CLANG_CXX_LIBRARY = "libc++";
+				CLANG_ENABLE_MODULES = YES;
+				CLANG_ENABLE_OBJC_ARC = YES;
+				CLANG_WARN_BLOCK_CAPTURE_AUTORELEASING = YES;
+				CLANG_WARN_BOOL_CONVERSION = YES;
+				CLANG_WARN_COMMA = YES;
+				CLANG_WARN_CONSTANT_CONVERSION = YES;
+				CLANG_WARN_DIRECT_OBJC_ISA_USAGE = YES_ERROR;
+				CLANG_WARN_EMPTY_BODY = YES;
+				CLANG_WARN_ENUM_CONVERSION = YES;
+				CLANG_WARN_INFINITE_RECURSION = YES;
+				CLANG_WARN_INT_CONVERSION = YES;
+				CLANG_WARN_NON_LITERAL_NULL_CONVERSION = YES;
+				CLANG_WARN_OBJC_LITERAL_CONVERSION = YES;
+				CLANG_WARN_OBJC_ROOT_CLASS = YES_ERROR;
+				CLANG_WARN_RANGE_LOOP_ANALYSIS = YES;
+				CLANG_WARN_STRICT_PROTOTYPES = YES;
+				CLANG_WARN_SUSPICIOUS_MOVE = YES;
+				CLANG_WARN_UNREACHABLE_CODE = YES;
+				CLANG_WARN__DUPLICATE_METHOD_MATCH = YES;
+				"CODE_SIGN_IDENTITY[sdk=iphoneos*]" = "iPhone Developer";
+				COPY_PHASE_STRIP = NO;
+				DEBUG_INFORMATION_FORMAT = "dwarf-with-dsym";
+				ENABLE_NS_ASSERTIONS = NO;
+				ENABLE_STRICT_OBJC_MSGSEND = YES;
+				GCC_C_LANGUAGE_STANDARD = gnu99;
+				GCC_NO_COMMON_BLOCKS = YES;
+				GCC_WARN_64_TO_32_BIT_CONVERSION = YES;
+				GCC_WARN_ABOUT_RETURN_TYPE = YES_ERROR;
+				GCC_WARN_UNDECLARED_SELECTOR = YES;
+				GCC_WARN_UNINITIALIZED_AUTOS = YES_AGGRESSIVE;
+				GCC_WARN_UNUSED_FUNCTION = YES;
+				GCC_WARN_UNUSED_VARIABLE = YES;
+				IPHONEOS_DEPLOYMENT_TARGET = 8.0;
+				MTL_ENABLE_DEBUG_INFO = NO;
+				SDKROOT = iphoneos;
+				TARGETED_DEVICE_FAMILY = "1,2";
+				VALIDATE_PRODUCT = YES;
+			};
+			name = Release;
+		};
+		97C147061CF9000F007C117D /* Debug */ = {
+			isa = XCBuildConfiguration;
+			baseConfigurationReference = 9740EEB21CF90195004384FC /* Debug.xcconfig */;
+			buildSettings = {
+				ASSETCATALOG_COMPILER_APPICON_NAME = AppIcon;
+				CURRENT_PROJECT_VERSION = "$(FLUTTER_BUILD_NUMBER)";
+				ENABLE_BITCODE = NO;
+				FRAMEWORK_SEARCH_PATHS = (
+					"$(inherited)",
+					"$(PROJECT_DIR)/Flutter",
+				);
+				INFOPLIST_FILE = Runner/Info.plist;
+				LD_RUNPATH_SEARCH_PATHS = "$(inherited) @executable_path/Frameworks";
+				LIBRARY_SEARCH_PATHS = (
+					"$(inherited)",
+					"$(PROJECT_DIR)/Flutter",
+				);
+				PRODUCT_BUNDLE_IDENTIFIER = com.arthenica.flutter.ffmpeg.flutterFfmpegExample;
+				PRODUCT_NAME = "$(TARGET_NAME)";
+				VERSIONING_SYSTEM = "apple-generic";
+			};
+			name = Debug;
+		};
+		97C147071CF9000F007C117D /* Release */ = {
+			isa = XCBuildConfiguration;
+			baseConfigurationReference = 7AFA3C8E1D35360C0083082E /* Release.xcconfig */;
+			buildSettings = {
+				ASSETCATALOG_COMPILER_APPICON_NAME = AppIcon;
+				CURRENT_PROJECT_VERSION = "$(FLUTTER_BUILD_NUMBER)";
+				ENABLE_BITCODE = NO;
+				FRAMEWORK_SEARCH_PATHS = (
+					"$(inherited)",
+					"$(PROJECT_DIR)/Flutter",
+				);
+				INFOPLIST_FILE = Runner/Info.plist;
+				LD_RUNPATH_SEARCH_PATHS = "$(inherited) @executable_path/Frameworks";
+				LIBRARY_SEARCH_PATHS = (
+					"$(inherited)",
+					"$(PROJECT_DIR)/Flutter",
+				);
+				PRODUCT_BUNDLE_IDENTIFIER = com.arthenica.flutter.ffmpeg.flutterFfmpegExample;
+				PRODUCT_NAME = "$(TARGET_NAME)";
+				VERSIONING_SYSTEM = "apple-generic";
+			};
+			name = Release;
+		};
+/* End XCBuildConfiguration section */
+
+/* Begin XCConfigurationList section */
+		97C146E91CF9000F007C117D /* Build configuration list for PBXProject "Runner" */ = {
+			isa = XCConfigurationList;
+			buildConfigurations = (
+				97C147031CF9000F007C117D /* Debug */,
+				97C147041CF9000F007C117D /* Release */,
+				249021D3217E4FDB00AE95B9 /* Profile */,
+			);
+			defaultConfigurationIsVisible = 0;
+			defaultConfigurationName = Release;
+		};
+		97C147051CF9000F007C117D /* Build configuration list for PBXNativeTarget "Runner" */ = {
+			isa = XCConfigurationList;
+			buildConfigurations = (
+				97C147061CF9000F007C117D /* Debug */,
+				97C147071CF9000F007C117D /* Release */,
+				249021D4217E4FDB00AE95B9 /* Profile */,
+			);
+			defaultConfigurationIsVisible = 0;
+			defaultConfigurationName = Release;
+		};
+/* End XCConfigurationList section */
+	};
+	rootObject = 97C146E61CF9000F007C117D /* Project object */;
+}

+ 7 - 0
example/ios/Runner.xcodeproj/project.xcworkspace/contents.xcworkspacedata

@@ -0,0 +1,7 @@
+<?xml version="1.0" encoding="UTF-8"?>
+<Workspace
+   version = "1.0">
+   <FileRef
+      location = "group:Runner.xcodeproj">
+   </FileRef>
+</Workspace>

+ 93 - 0
example/ios/Runner.xcodeproj/xcshareddata/xcschemes/Runner.xcscheme

@@ -0,0 +1,93 @@
+<?xml version="1.0" encoding="UTF-8"?>
+<Scheme
+   LastUpgradeVersion = "0910"
+   version = "1.3">
+   <BuildAction
+      parallelizeBuildables = "YES"
+      buildImplicitDependencies = "YES">
+      <BuildActionEntries>
+         <BuildActionEntry
+            buildForTesting = "YES"
+            buildForRunning = "YES"
+            buildForProfiling = "YES"
+            buildForArchiving = "YES"
+            buildForAnalyzing = "YES">
+            <BuildableReference
+               BuildableIdentifier = "primary"
+               BlueprintIdentifier = "97C146ED1CF9000F007C117D"
+               BuildableName = "Runner.app"
+               BlueprintName = "Runner"
+               ReferencedContainer = "container:Runner.xcodeproj">
+            </BuildableReference>
+         </BuildActionEntry>
+      </BuildActionEntries>
+   </BuildAction>
+   <TestAction
+      buildConfiguration = "Debug"
+      selectedDebuggerIdentifier = "Xcode.DebuggerFoundation.Debugger.LLDB"
+      selectedLauncherIdentifier = "Xcode.DebuggerFoundation.Launcher.LLDB"
+      language = ""
+      shouldUseLaunchSchemeArgsEnv = "YES">
+      <Testables>
+      </Testables>
+      <MacroExpansion>
+         <BuildableReference
+            BuildableIdentifier = "primary"
+            BlueprintIdentifier = "97C146ED1CF9000F007C117D"
+            BuildableName = "Runner.app"
+            BlueprintName = "Runner"
+            ReferencedContainer = "container:Runner.xcodeproj">
+         </BuildableReference>
+      </MacroExpansion>
+      <AdditionalOptions>
+      </AdditionalOptions>
+   </TestAction>
+   <LaunchAction
+      buildConfiguration = "Debug"
+      selectedDebuggerIdentifier = "Xcode.DebuggerFoundation.Debugger.LLDB"
+      selectedLauncherIdentifier = "Xcode.DebuggerFoundation.Launcher.LLDB"
+      language = ""
+      launchStyle = "0"
+      useCustomWorkingDirectory = "NO"
+      ignoresPersistentStateOnLaunch = "NO"
+      debugDocumentVersioning = "YES"
+      debugServiceExtension = "internal"
+      allowLocationSimulation = "YES">
+      <BuildableProductRunnable
+         runnableDebuggingMode = "0">
+         <BuildableReference
+            BuildableIdentifier = "primary"
+            BlueprintIdentifier = "97C146ED1CF9000F007C117D"
+            BuildableName = "Runner.app"
+            BlueprintName = "Runner"
+            ReferencedContainer = "container:Runner.xcodeproj">
+         </BuildableReference>
+      </BuildableProductRunnable>
+      <AdditionalOptions>
+      </AdditionalOptions>
+   </LaunchAction>
+   <ProfileAction
+      buildConfiguration = "Profile"
+      shouldUseLaunchSchemeArgsEnv = "YES"
+      savedToolIdentifier = ""
+      useCustomWorkingDirectory = "NO"
+      debugDocumentVersioning = "YES">
+      <BuildableProductRunnable
+         runnableDebuggingMode = "0">
+         <BuildableReference
+            BuildableIdentifier = "primary"
+            BlueprintIdentifier = "97C146ED1CF9000F007C117D"
+            BuildableName = "Runner.app"
+            BlueprintName = "Runner"
+            ReferencedContainer = "container:Runner.xcodeproj">
+         </BuildableReference>
+      </BuildableProductRunnable>
+   </ProfileAction>
+   <AnalyzeAction
+      buildConfiguration = "Debug">
+   </AnalyzeAction>
+   <ArchiveAction
+      buildConfiguration = "Release"
+      revealArchiveInOrganizer = "YES">
+   </ArchiveAction>
+</Scheme>

+ 10 - 0
example/ios/Runner.xcworkspace/contents.xcworkspacedata

@@ -0,0 +1,10 @@
+<?xml version="1.0" encoding="UTF-8"?>
+<Workspace
+   version = "1.0">
+   <FileRef
+      location = "group:Runner.xcodeproj">
+   </FileRef>
+   <FileRef
+      location = "group:Pods/Pods.xcodeproj">
+   </FileRef>
+</Workspace>

+ 8 - 0
example/ios/Runner.xcworkspace/xcshareddata/IDEWorkspaceChecks.plist

@@ -0,0 +1,8 @@
+<?xml version="1.0" encoding="UTF-8"?>
+<!DOCTYPE plist PUBLIC "-//Apple//DTD PLIST 1.0//EN" "http://www.apple.com/DTDs/PropertyList-1.0.dtd">
+<plist version="1.0">
+<dict>
+	<key>IDEDidComputeMac32BitWarning</key>
+	<true/>
+</dict>
+</plist>

+ 8 - 0
example/ios/Runner.xcworkspace/xcshareddata/WorkspaceSettings.xcsettings

@@ -0,0 +1,8 @@
+<?xml version="1.0" encoding="UTF-8"?>
+<!DOCTYPE plist PUBLIC "-//Apple//DTD PLIST 1.0//EN" "http://www.apple.com/DTDs/PropertyList-1.0.dtd">
+<plist version="1.0">
+<dict>
+	<key>BuildSystemType</key>
+	<string>Original</string>
+</dict>
+</plist>

+ 6 - 0
example/ios/Runner/AppDelegate.h

@@ -0,0 +1,6 @@
+#import <Flutter/Flutter.h>
+#import <UIKit/UIKit.h>
+
+@interface AppDelegate : FlutterAppDelegate
+
+@end

+ 13 - 0
example/ios/Runner/AppDelegate.m

@@ -0,0 +1,13 @@
+#include "AppDelegate.h"
+#include "GeneratedPluginRegistrant.h"
+
+@implementation AppDelegate
+
+- (BOOL)application:(UIApplication *)application
+    didFinishLaunchingWithOptions:(NSDictionary *)launchOptions {
+  [GeneratedPluginRegistrant registerWithRegistry:self];
+  // Override point for customization after application launch.
+  return [super application:application didFinishLaunchingWithOptions:launchOptions];
+}
+
+@end

+ 122 - 0
example/ios/Runner/Assets.xcassets/AppIcon.appiconset/Contents.json

@@ -0,0 +1,122 @@
+{
+  "images" : [
+    {
+      "size" : "20x20",
+      "idiom" : "iphone",
+      "filename" : "Icon-App-20x20@2x.png",
+      "scale" : "2x"
+    },
+    {
+      "size" : "20x20",
+      "idiom" : "iphone",
+      "filename" : "Icon-App-20x20@3x.png",
+      "scale" : "3x"
+    },
+    {
+      "size" : "29x29",
+      "idiom" : "iphone",
+      "filename" : "Icon-App-29x29@1x.png",
+      "scale" : "1x"
+    },
+    {
+      "size" : "29x29",
+      "idiom" : "iphone",
+      "filename" : "Icon-App-29x29@2x.png",
+      "scale" : "2x"
+    },
+    {
+      "size" : "29x29",
+      "idiom" : "iphone",
+      "filename" : "Icon-App-29x29@3x.png",
+      "scale" : "3x"
+    },
+    {
+      "size" : "40x40",
+      "idiom" : "iphone",
+      "filename" : "Icon-App-40x40@2x.png",
+      "scale" : "2x"
+    },
+    {
+      "size" : "40x40",
+      "idiom" : "iphone",
+      "filename" : "Icon-App-40x40@3x.png",
+      "scale" : "3x"
+    },
+    {
+      "size" : "60x60",
+      "idiom" : "iphone",
+      "filename" : "Icon-App-60x60@2x.png",
+      "scale" : "2x"
+    },
+    {
+      "size" : "60x60",
+      "idiom" : "iphone",
+      "filename" : "Icon-App-60x60@3x.png",
+      "scale" : "3x"
+    },
+    {
+      "size" : "20x20",
+      "idiom" : "ipad",
+      "filename" : "Icon-App-20x20@1x.png",
+      "scale" : "1x"
+    },
+    {
+      "size" : "20x20",
+      "idiom" : "ipad",
+      "filename" : "Icon-App-20x20@2x.png",
+      "scale" : "2x"
+    },
+    {
+      "size" : "29x29",
+      "idiom" : "ipad",
+      "filename" : "Icon-App-29x29@1x.png",
+      "scale" : "1x"
+    },
+    {
+      "size" : "29x29",
+      "idiom" : "ipad",
+      "filename" : "Icon-App-29x29@2x.png",
+      "scale" : "2x"
+    },
+    {
+      "size" : "40x40",
+      "idiom" : "ipad",
+      "filename" : "Icon-App-40x40@1x.png",
+      "scale" : "1x"
+    },
+    {
+      "size" : "40x40",
+      "idiom" : "ipad",
+      "filename" : "Icon-App-40x40@2x.png",
+      "scale" : "2x"
+    },
+    {
+      "size" : "76x76",
+      "idiom" : "ipad",
+      "filename" : "Icon-App-76x76@1x.png",
+      "scale" : "1x"
+    },
+    {
+      "size" : "76x76",
+      "idiom" : "ipad",
+      "filename" : "Icon-App-76x76@2x.png",
+      "scale" : "2x"
+    },
+    {
+      "size" : "83.5x83.5",
+      "idiom" : "ipad",
+      "filename" : "Icon-App-83.5x83.5@2x.png",
+      "scale" : "2x"
+    },
+    {
+      "size" : "1024x1024",
+      "idiom" : "ios-marketing",
+      "filename" : "Icon-App-1024x1024@1x.png",
+      "scale" : "1x"
+    }
+  ],
+  "info" : {
+    "version" : 1,
+    "author" : "xcode"
+  }
+}

BIN
example/ios/Runner/Assets.xcassets/AppIcon.appiconset/Icon-App-1024x1024@1x.png


BIN
example/ios/Runner/Assets.xcassets/AppIcon.appiconset/Icon-App-20x20@1x.png


BIN
example/ios/Runner/Assets.xcassets/AppIcon.appiconset/Icon-App-20x20@2x.png


BIN
example/ios/Runner/Assets.xcassets/AppIcon.appiconset/Icon-App-20x20@3x.png


BIN
example/ios/Runner/Assets.xcassets/AppIcon.appiconset/Icon-App-29x29@1x.png


BIN
example/ios/Runner/Assets.xcassets/AppIcon.appiconset/Icon-App-29x29@2x.png


BIN
example/ios/Runner/Assets.xcassets/AppIcon.appiconset/Icon-App-29x29@3x.png


BIN
example/ios/Runner/Assets.xcassets/AppIcon.appiconset/Icon-App-40x40@1x.png


BIN
example/ios/Runner/Assets.xcassets/AppIcon.appiconset/Icon-App-40x40@2x.png


BIN
example/ios/Runner/Assets.xcassets/AppIcon.appiconset/Icon-App-40x40@3x.png


BIN
example/ios/Runner/Assets.xcassets/AppIcon.appiconset/Icon-App-60x60@2x.png


BIN
example/ios/Runner/Assets.xcassets/AppIcon.appiconset/Icon-App-60x60@3x.png


BIN
example/ios/Runner/Assets.xcassets/AppIcon.appiconset/Icon-App-76x76@1x.png


BIN
example/ios/Runner/Assets.xcassets/AppIcon.appiconset/Icon-App-76x76@2x.png


BIN
example/ios/Runner/Assets.xcassets/AppIcon.appiconset/Icon-App-83.5x83.5@2x.png


+ 23 - 0
example/ios/Runner/Assets.xcassets/LaunchImage.imageset/Contents.json

@@ -0,0 +1,23 @@
+{
+  "images" : [
+    {
+      "idiom" : "universal",
+      "filename" : "LaunchImage.png",
+      "scale" : "1x"
+    },
+    {
+      "idiom" : "universal",
+      "filename" : "LaunchImage@2x.png",
+      "scale" : "2x"
+    },
+    {
+      "idiom" : "universal",
+      "filename" : "LaunchImage@3x.png",
+      "scale" : "3x"
+    }
+  ],
+  "info" : {
+    "version" : 1,
+    "author" : "xcode"
+  }
+}

BIN
example/ios/Runner/Assets.xcassets/LaunchImage.imageset/LaunchImage.png


BIN
example/ios/Runner/Assets.xcassets/LaunchImage.imageset/LaunchImage@2x.png


BIN
example/ios/Runner/Assets.xcassets/LaunchImage.imageset/LaunchImage@3x.png


+ 5 - 0
example/ios/Runner/Assets.xcassets/LaunchImage.imageset/README.md

@@ -0,0 +1,5 @@
+# Launch Screen Assets
+
+You can customize the launch screen with your own desired assets by replacing the image files in this directory.
+
+You can also do it by opening your Flutter project's Xcode project with `open ios/Runner.xcworkspace`, selecting `Runner/Assets.xcassets` in the Project Navigator and dropping in the desired images.

+ 37 - 0
example/ios/Runner/Base.lproj/LaunchScreen.storyboard

@@ -0,0 +1,37 @@
+<?xml version="1.0" encoding="UTF-8" standalone="no"?>
+<document type="com.apple.InterfaceBuilder3.CocoaTouch.Storyboard.XIB" version="3.0" toolsVersion="12121" systemVersion="16G29" targetRuntime="iOS.CocoaTouch" propertyAccessControl="none" useAutolayout="YES" launchScreen="YES" colorMatched="YES" initialViewController="01J-lp-oVM">
+    <dependencies>
+        <deployment identifier="iOS"/>
+        <plugIn identifier="com.apple.InterfaceBuilder.IBCocoaTouchPlugin" version="12089"/>
+    </dependencies>
+    <scenes>
+        <!--View Controller-->
+        <scene sceneID="EHf-IW-A2E">
+            <objects>
+                <viewController id="01J-lp-oVM" sceneMemberID="viewController">
+                    <layoutGuides>
+                        <viewControllerLayoutGuide type="top" id="Ydg-fD-yQy"/>
+                        <viewControllerLayoutGuide type="bottom" id="xbc-2k-c8Z"/>
+                    </layoutGuides>
+                    <view key="view" contentMode="scaleToFill" id="Ze5-6b-2t3">
+                        <autoresizingMask key="autoresizingMask" widthSizable="YES" heightSizable="YES"/>
+                        <subviews>
+                            <imageView opaque="NO" clipsSubviews="YES" multipleTouchEnabled="YES" contentMode="center" image="LaunchImage" translatesAutoresizingMaskIntoConstraints="NO" id="YRO-k0-Ey4">
+                            </imageView>
+                        </subviews>
+                        <color key="backgroundColor" red="1" green="1" blue="1" alpha="1" colorSpace="custom" customColorSpace="sRGB"/>
+                        <constraints>
+                            <constraint firstItem="YRO-k0-Ey4" firstAttribute="centerX" secondItem="Ze5-6b-2t3" secondAttribute="centerX" id="1a2-6s-vTC"/>
+                            <constraint firstItem="YRO-k0-Ey4" firstAttribute="centerY" secondItem="Ze5-6b-2t3" secondAttribute="centerY" id="4X2-HB-R7a"/>
+                        </constraints>
+                    </view>
+                </viewController>
+                <placeholder placeholderIdentifier="IBFirstResponder" id="iYj-Kq-Ea1" userLabel="First Responder" sceneMemberID="firstResponder"/>
+            </objects>
+            <point key="canvasLocation" x="53" y="375"/>
+        </scene>
+    </scenes>
+    <resources>
+        <image name="LaunchImage" width="168" height="185"/>
+    </resources>
+</document>

+ 26 - 0
example/ios/Runner/Base.lproj/Main.storyboard

@@ -0,0 +1,26 @@
+<?xml version="1.0" encoding="UTF-8" standalone="no"?>
+<document type="com.apple.InterfaceBuilder3.CocoaTouch.Storyboard.XIB" version="3.0" toolsVersion="10117" systemVersion="15F34" targetRuntime="iOS.CocoaTouch" propertyAccessControl="none" useAutolayout="YES" useTraitCollections="YES" initialViewController="BYZ-38-t0r">
+    <dependencies>
+        <deployment identifier="iOS"/>
+        <plugIn identifier="com.apple.InterfaceBuilder.IBCocoaTouchPlugin" version="10085"/>
+    </dependencies>
+    <scenes>
+        <!--Flutter View Controller-->
+        <scene sceneID="tne-QT-ifu">
+            <objects>
+                <viewController id="BYZ-38-t0r" customClass="FlutterViewController" sceneMemberID="viewController">
+                    <layoutGuides>
+                        <viewControllerLayoutGuide type="top" id="y3c-jy-aDJ"/>
+                        <viewControllerLayoutGuide type="bottom" id="wfy-db-euE"/>
+                    </layoutGuides>
+                    <view key="view" contentMode="scaleToFill" id="8bC-Xf-vdC">
+                        <rect key="frame" x="0.0" y="0.0" width="600" height="600"/>
+                        <autoresizingMask key="autoresizingMask" widthSizable="YES" heightSizable="YES"/>
+                        <color key="backgroundColor" white="1" alpha="1" colorSpace="custom" customColorSpace="calibratedWhite"/>
+                    </view>
+                </viewController>
+                <placeholder placeholderIdentifier="IBFirstResponder" id="dkx-z0-nzr" sceneMemberID="firstResponder"/>
+            </objects>
+        </scene>
+    </scenes>
+</document>

+ 45 - 0
example/ios/Runner/Info.plist

@@ -0,0 +1,45 @@
+<?xml version="1.0" encoding="UTF-8"?>
+<!DOCTYPE plist PUBLIC "-//Apple//DTD PLIST 1.0//EN" "http://www.apple.com/DTDs/PropertyList-1.0.dtd">
+<plist version="1.0">
+<dict>
+	<key>CFBundleDevelopmentRegion</key>
+	<string>en</string>
+	<key>CFBundleExecutable</key>
+	<string>$(EXECUTABLE_NAME)</string>
+	<key>CFBundleIdentifier</key>
+	<string>$(PRODUCT_BUNDLE_IDENTIFIER)</string>
+	<key>CFBundleInfoDictionaryVersion</key>
+	<string>6.0</string>
+	<key>CFBundleName</key>
+	<string>flutter_ffmpeg_example</string>
+	<key>CFBundlePackageType</key>
+	<string>APPL</string>
+	<key>CFBundleShortVersionString</key>
+	<string>$(FLUTTER_BUILD_NAME)</string>
+	<key>CFBundleSignature</key>
+	<string>????</string>
+	<key>CFBundleVersion</key>
+	<string>$(FLUTTER_BUILD_NUMBER)</string>
+	<key>LSRequiresIPhoneOS</key>
+	<true/>
+	<key>UILaunchStoryboardName</key>
+	<string>LaunchScreen</string>
+	<key>UIMainStoryboardFile</key>
+	<string>Main</string>
+	<key>UISupportedInterfaceOrientations</key>
+	<array>
+		<string>UIInterfaceOrientationPortrait</string>
+		<string>UIInterfaceOrientationLandscapeLeft</string>
+		<string>UIInterfaceOrientationLandscapeRight</string>
+	</array>
+	<key>UISupportedInterfaceOrientations~ipad</key>
+	<array>
+		<string>UIInterfaceOrientationPortrait</string>
+		<string>UIInterfaceOrientationPortraitUpsideDown</string>
+		<string>UIInterfaceOrientationLandscapeLeft</string>
+		<string>UIInterfaceOrientationLandscapeRight</string>
+	</array>
+	<key>UIViewControllerBasedStatusBarAppearance</key>
+	<false/>
+</dict>
+</plist>

+ 9 - 0
example/ios/Runner/main.m

@@ -0,0 +1,9 @@
+#import <Flutter/Flutter.h>
+#import <UIKit/UIKit.h>
+#import "AppDelegate.h"
+
+int main(int argc, char* argv[]) {
+  @autoreleasepool {
+    return UIApplicationMain(argc, argv, nil, NSStringFromClass([AppDelegate class]));
+  }
+}

+ 41 - 0
example/lib/flutter_ffmpeg_test_app.dart

@@ -0,0 +1,41 @@
+import 'package:flutter/material.dart';
+import 'package:flutter_ffmpeg_example/flutter_ffmpeg_test_app_state.dart';
+
+void main() => runApp(FlutterFFmpegTestApp());
+
+class FlutterFFmpegTestApp extends StatelessWidget {
+  @override
+  Widget build(BuildContext context) {
+    return MaterialApp(
+      theme: ThemeData(
+        primaryColor: Color(0xFFF46842),
+      ),
+      home: MainPage(),
+    );
+  }
+}
+
+class MainPage extends StatefulWidget {
+  @override
+  FlutterFFmpegTestAppState createState() => new FlutterFFmpegTestAppState();
+}
+
+class DecoratedTabBar extends StatelessWidget implements PreferredSizeWidget {
+  DecoratedTabBar({@required this.tabBar, @required this.decoration});
+
+  final TabBar tabBar;
+  final BoxDecoration decoration;
+
+  @override
+  Size get preferredSize => tabBar.preferredSize;
+
+  @override
+  Widget build(BuildContext context) {
+    return Stack(
+      children: [
+        Positioned.fill(child: Container(decoration: decoration)),
+        tabBar,
+      ],
+    );
+  }
+}

+ 547 - 0
example/lib/flutter_ffmpeg_test_app_state.dart

@@ -0,0 +1,547 @@
+import 'dart:async';
+
+import 'package:flutter/material.dart';
+import 'package:flutter_ffmpeg/flutter_ffmpeg.dart';
+import 'package:flutter_ffmpeg/log_level.dart';
+import 'package:flutter_ffmpeg_example/flutter_ffmpeg_test_app.dart';
+import 'package:flutter_ffmpeg_example/video_util.dart';
+
+class FlutterFFmpegTestAppState extends State<MainPage> with TickerProviderStateMixin {
+  static const String ASSET_1 = "1.jpg";
+  static const String ASSET_2 = "2.jpg";
+  static const String ASSET_3 = "3.jpg";
+
+  final FlutterFFmpeg _flutterFFmpeg = new FlutterFFmpeg();
+
+  TextEditingController _commandController;
+  TabController _controller;
+  String _commandOutput;
+  String _encodeOutput;
+  String _currentCodec;
+  List<DropdownMenuItem<String>> _codecDropDownMenuItems;
+
+  @override
+  void initState() {
+    super.initState();
+
+    _commandController = TextEditingController();
+    _controller = TabController(length: 2, vsync: this);
+    _commandOutput = "";
+    _encodeOutput = "";
+
+    _codecDropDownMenuItems = _getCodecDropDownMenuItems();
+    _currentCodec = _codecDropDownMenuItems[0].value;
+
+    startupTests();
+
+    prepareAssets();
+  }
+
+  void startupTests() {
+    getFFmpegVersion().then((version) => print("FFmpeg version: $version"));
+    getPlatform().then((platform) => print("Platform: $platform"));
+    getLogLevel().then((level) => print("Old log level: " + LogLevel.levelToString(level)));
+    setLogLevel(LogLevel.AV_LOG_INFO);
+    getLogLevel().then((level) => print("New log level: " + LogLevel.levelToString(level)));
+    getPackageName().then((packageName) => print("Package name: $packageName"));
+    getExternalLibraries().then((packageList) {
+      packageList.forEach((value) => print("External library: $value"));
+    });
+  }
+
+  void prepareAssets() {
+    VideoUtil.copyFileAssets('assets/pyramid.jpg', ASSET_1).then((path) => print('Loaded asset $path.'));
+    VideoUtil.copyFileAssets('assets/colosseum.jpg', ASSET_2).then((path) => print('Loaded asset $path.'));
+    VideoUtil.copyFileAssets('assets/tajmahal.jpg', ASSET_3).then((path) => print('Loaded asset $path.'));
+  }
+
+  void testRunCommand() {
+    getLastReturnCode().then((rc) => print("Last rc: $rc"));
+    getLastCommandOutput().then((output) => print("Last command output: $output"));
+
+    print("Testing COMMAND.");
+
+    // ENABLE LOG CALLBACK ON EACH CALL
+    enableLogCallback(commandOutputLogCallback);
+    enableStatisticsCallback(statisticsCallback);
+
+    // CLEAR OUTPUT ON EACH EXECUTION
+    _commandOutput = "";
+
+    // COMMENT OPTIONAL TESTS
+    // VideoUtil.tempDirectory.then((tempDirectory) {
+    //    Map<String, String> mapNameMap = new Map();
+    //    mapNameMap["my_custom_font"] = "my custom font";
+    //    setFontDirectory(tempDirectory.path, null);
+    // });
+
+    VideoUtil.tempDirectory.then((tempDirectory) {
+      setFontconfigConfigurationPath(tempDirectory.path);
+    });
+
+    // disableRedirection();
+
+    // disableLogs();
+    // enableLogs();
+
+    // execute(_commandController.text).then((rc) => print("FFmpeg process exited with rc $rc"));
+    // executeWithDelimiter(_commandController.text, "_").then((rc) => print("FFmpeg process exited with rc $rc") );
+    executeWithArguments(_commandController.text.split(" ")).then((rc) => print("FFmpeg process exited with rc $rc"));
+
+    setState(() {});
+  }
+
+  void testGetMediaInformation(String mediaPath) {
+    print("Testing Get Media Information.");
+
+    // ENABLE LOG CALLBACK ON EACH CALL
+    enableLogCallback(commandOutputLogCallback);
+    enableStatisticsCallback(null);
+
+    // CLEAR OUTPUT ON EACH EXECUTION
+    _commandOutput = "";
+
+    VideoUtil.assetPath(mediaPath).then((image1Path) {
+      getMediaInformation(image1Path).then((info) {
+        print('Media Information');
+
+        print('Path: ${info['path']}');
+        print('Format: ${info['format']}');
+        print('Duration: ${info['duration']}');
+        print('Start time: ${info['startTime']}');
+        print('Bitrate: ${info['bitrate']}');
+
+        if (info['streams'] != null) {
+          final streamsInfoArray = info['streams'];
+
+          if (streamsInfoArray.length > 0) {
+            for (var streamsInfo in streamsInfoArray) {
+              print('Stream id: ${streamsInfo['index']}');
+              print('Stream type: ${streamsInfo['type']}');
+              print('Stream codec: ${streamsInfo['codec']}');
+              print('Stream full codec: ${streamsInfo['fullCodec']}');
+              print('Stream format: ${streamsInfo['format']}');
+              print('Stream full format: ${streamsInfo['fullFormat']}');
+              print('Stream width: ${streamsInfo['width']}');
+              print('Stream height: ${streamsInfo['height']}');
+              print('Stream bitrate: ${streamsInfo['bitrate']}');
+              print('Stream sample rate: ${streamsInfo['sampleRate']}');
+              print('Stream sample format: ${streamsInfo['sampleFormat']}');
+              print('Stream channel layout: ${streamsInfo['channelLayout']}');
+              print('Stream sar: ${streamsInfo['sampleAspectRatio']}');
+              print('Stream dar: ${streamsInfo['displayAspectRatio']}');
+              print('Stream average frame rate: ${streamsInfo['averageFrameRate']}');
+              print('Stream real frame rate: ${streamsInfo['realFrameRate']}');
+              print('Stream time base: ${streamsInfo['timeBase']}');
+              print('Stream codec time base: ${streamsInfo['codecTimeBase']}');
+            }
+          }
+        }
+      });
+    });
+
+    setState(() {});
+  }
+
+  void testEncodeVideo() {
+    print("Testing VIDEO.");
+
+    // ENABLE LOG CALLBACK ON EACH CALL
+    enableLogCallback(encodeOutputLogCallback);
+    enableStatisticsCallback(statisticsCallback);
+
+    // CLEAR OUTPUT ON EACH EXECUTION
+    _encodeOutput = "";
+
+    disableStatistics();
+    enableStatistics();
+
+    VideoUtil.assetPath(ASSET_1).then((image1Path) {
+      VideoUtil.assetPath(ASSET_2).then((image2Path) {
+        VideoUtil.assetPath(ASSET_3).then((image3Path) {
+          final String videoPath = getVideoPath();
+          final String customOptions = getCustomEncodingOptions();
+          final String ffmpegCodec = getFFmpegCodecName();
+
+          VideoUtil.assetPath(videoPath).then((fullVideoPath) {
+            execute(VideoUtil.generateEncodeVideoScript(image1Path, image2Path, image3Path, fullVideoPath, ffmpegCodec, customOptions)).then((rc) {
+              if (rc == 0) {
+                testGetMediaInformation(fullVideoPath);
+              }
+            });
+
+            // COMMENT OPTIONAL TESTS
+            // execute(VideoUtil.generateEncodeVideoScript(image1Path, image2Path, image3Path, videoPath, _currentCodec, "")).timeout(Duration(milliseconds: 1300), onTimeout: () { cancel(); });
+
+            // resetStatistics();
+
+            getLastReceivedStatistics().then((lastStatistics) {
+              if (lastStatistics == null) {
+                print('No last statistics');
+              } else {
+                print('Last statistics');
+
+                int time = lastStatistics['time'];
+                int size = lastStatistics['size'];
+
+                double bitrate = _doublePrecision(lastStatistics['bitrate'], 2);
+                double speed = _doublePrecision(lastStatistics['speed'], 2);
+                int videoFrameNumber = lastStatistics['videoFrameNumber'];
+                double videoQuality = _doublePrecision(lastStatistics['videoQuality'], 2);
+                double videoFps = _doublePrecision(lastStatistics['videoFps'], 2);
+
+                statisticsCallback(time, size, bitrate, speed, videoFrameNumber, videoQuality, videoFps);
+              }
+            });
+          });
+        });
+      });
+    });
+
+    setState(() {});
+  }
+
+  void commandOutputLogCallback(int level, String message) {
+    _commandOutput += message;
+    setState(() {});
+  }
+
+  void encodeOutputLogCallback(int level, String message) {
+    _encodeOutput += message;
+    setState(() {});
+  }
+
+  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");
+  }
+
+  Future<String> getFFmpegVersion() async {
+    return await _flutterFFmpeg.getFFmpegVersion();
+  }
+
+  Future<String> getPlatform() async {
+    return await _flutterFFmpeg.getPlatform();
+  }
+
+  Future<int> executeWithArguments(List arguments) async {
+    return await _flutterFFmpeg.executeWithArguments(arguments);
+  }
+
+  Future<int> execute(String command) async {
+    return await _flutterFFmpeg.execute(command);
+  }
+
+  Future<int> executeWithDelimiter(String command, String delimiter) async {
+    return await _flutterFFmpeg.execute(command, delimiter);
+  }
+
+  Future<void> cancel() async {
+    return await _flutterFFmpeg.cancel();
+  }
+
+  Future<void> disableRedirection() async {
+    return await _flutterFFmpeg.disableRedirection();
+  }
+
+  Future<int> getLogLevel() async {
+    return await _flutterFFmpeg.getLogLevel();
+  }
+
+  Future<void> setLogLevel(int logLevel) async {
+    return await _flutterFFmpeg.setLogLevel(logLevel);
+  }
+
+  Future<void> enableLogs() async {
+    return await _flutterFFmpeg.enableLogs();
+  }
+
+  Future<void> disableLogs() async {
+    return await _flutterFFmpeg.disableLogs();
+  }
+
+  Future<void> enableStatistics() async {
+    return await _flutterFFmpeg.enableStatistics();
+  }
+
+  Future<void> disableStatistics() async {
+    return await _flutterFFmpeg.disableStatistics();
+  }
+
+  Future<void> enableLogCallback(Function(int level, String message) logCallback) async {
+    await _flutterFFmpeg.enableLogCallback(logCallback);
+  }
+
+  Future<void> enableStatisticsCallback(Function(int time, int size, double bitrate, double speed, int videoFrameNumber, double videoQuality, double videoFps) statisticsCallback) async {
+    await _flutterFFmpeg.enableStatisticsCallback(statisticsCallback);
+  }
+
+  Future<Map<dynamic, dynamic>> getLastReceivedStatistics() async {
+    return await _flutterFFmpeg.getLastReceivedStatistics();
+  }
+
+  Future<void> resetStatistics() async {
+    return await _flutterFFmpeg.resetStatistics();
+  }
+
+  Future<void> setFontconfigConfigurationPath(String path) async {
+    return await _flutterFFmpeg.setFontconfigConfigurationPath(path);
+  }
+
+  Future<void> setFontDirectory(String fontDirectory, Map<String, String> fontNameMap) async {
+    return await _flutterFFmpeg.setFontDirectory(fontDirectory, fontNameMap);
+  }
+
+  Future<String> getPackageName() async {
+    return await _flutterFFmpeg.getPackageName();
+  }
+
+  Future<List<dynamic>> getExternalLibraries() async {
+    return await _flutterFFmpeg.getExternalLibraries();
+  }
+
+  Future<int> getLastReturnCode() async {
+    return await _flutterFFmpeg.getLastReturnCode();
+  }
+
+  Future<String> getLastCommandOutput() async {
+    return await _flutterFFmpeg.getLastCommandOutput();
+  }
+
+  Future<Map<dynamic, dynamic>> getMediaInformation(String path) async {
+    return await _flutterFFmpeg.getMediaInformation(path);
+  }
+
+  void _changedCodec(String selectedCodec) {
+    setState(() {
+      _currentCodec = selectedCodec;
+    });
+  }
+
+  String getFFmpegCodecName() {
+    String ffmpegCodec = _currentCodec;
+
+    // VIDEO CODEC MENU HAS BASIC NAMES, FFMPEG NEEDS LONGER LIBRARY NAMES.
+    if (ffmpegCodec == "h264") {
+      ffmpegCodec = "libx264";
+    } else if (ffmpegCodec == "x265") {
+      ffmpegCodec = "libx265";
+    } else if (ffmpegCodec == "xvid") {
+      ffmpegCodec = "libxvid";
+    } else if (ffmpegCodec == "vp8") {
+      ffmpegCodec = "libvpx";
+    } else if (ffmpegCodec == "vp9") {
+      ffmpegCodec = "libvpx-vp9";
+    }
+
+    return ffmpegCodec;
+  }
+
+  String getVideoPath() {
+    String ffmpegCodec = _currentCodec;
+
+    String videoPath;
+    if ((ffmpegCodec == "vp8") || (ffmpegCodec == "vp9")) {
+      videoPath = "video.webm";
+    } else {
+      // mpeg4, x264, x265, xvid
+      videoPath = "video.mp4";
+    }
+
+    return videoPath;
+  }
+
+  String getCustomEncodingOptions() {
+    String videoCodec = _currentCodec;
+
+    if (videoCodec == "x265") {
+      return "-crf 28 -preset fast ";
+    } else if (videoCodec == "vp8") {
+      return "-b:v 1M -crf 10 ";
+    } else if (videoCodec == "vp9") {
+      return "-b:v 2M ";
+    } else {
+      return "";
+    }
+  }
+
+  List<DropdownMenuItem<String>> _getCodecDropDownMenuItems() {
+    List<DropdownMenuItem<String>> items = new List();
+
+    items.add(new DropdownMenuItem(value: "mpeg4", child: new Text("mpeg4")));
+    items.add(new DropdownMenuItem(value: "h264", child: new Text("h264")));
+    items.add(new DropdownMenuItem(value: "x265", child: new Text("x265")));
+    items.add(new DropdownMenuItem(value: "xvid", child: new Text("xvid")));
+    items.add(new DropdownMenuItem(value: "vp8", child: new Text("vp8")));
+    items.add(new DropdownMenuItem(value: "vp9", child: new Text("vp9")));
+
+    return items;
+  }
+
+  double _doublePrecision(double value, int precision) {
+    if (value == null) {
+      return 0;
+    } else {
+      return num.parse(value.toStringAsFixed(precision));
+    }
+  }
+
+  @override
+  Widget build(BuildContext context) {
+    return Scaffold(
+        appBar: AppBar(
+          title: Text('FlutterFFmpeg Test'),
+          centerTitle: true,
+        ),
+        bottomNavigationBar: Material(
+          child: DecoratedTabBar(
+            tabBar: TabBar(
+              tabs: <Tab>[
+                Tab(text: "COMMAND"),
+                Tab(
+                  text: "VIDEO",
+                )
+              ],
+              controller: _controller,
+              labelColor: Color(0xFF1e90ff),
+              unselectedLabelColor: Color(0xFF808080),
+            ),
+            decoration: BoxDecoration(
+              border: Border(
+                top: BorderSide(
+                  color: Color(0xFF808080),
+                  width: 1.0,
+                ),
+                bottom: BorderSide(
+                  width: 0.0,
+                ),
+              ),
+            ),
+          ),
+        ),
+        body: TabBarView(
+          children: <Widget>[
+            Column(
+              crossAxisAlignment: CrossAxisAlignment.center,
+              children: <Widget>[
+                Container(
+                  padding: const EdgeInsets.fromLTRB(20, 40, 20, 40),
+                  child: TextField(
+                    controller: _commandController,
+                    decoration: InputDecoration(
+                        border: const OutlineInputBorder(
+                          borderSide: const BorderSide(color: Color(0xFF3498DB)),
+                          borderRadius: const BorderRadius.all(
+                            const Radius.circular(5),
+                          ),
+                        ),
+                        focusedBorder: const OutlineInputBorder(
+                          borderSide: const BorderSide(color: Color(0xFF3498DB)),
+                          borderRadius: const BorderRadius.all(
+                            const Radius.circular(5),
+                          ),
+                        ),
+                        enabledBorder: const OutlineInputBorder(
+                          borderSide: const BorderSide(color: Color(0xFF3498DB)),
+                          borderRadius: const BorderRadius.all(
+                            const Radius.circular(5),
+                          ),
+                        ),
+                        contentPadding: EdgeInsets.fromLTRB(8, 12, 8, 12),
+                        hintStyle: new TextStyle(fontSize: 14, color: Colors.grey[400]),
+                        hintText: "Enter command"),
+                    style: new TextStyle(fontSize: 14, color: Colors.black),
+                  ),
+                ),
+                Container(
+                  padding: const EdgeInsets.only(bottom: 20),
+                  child: new InkWell(
+                    onTap: () => testRunCommand(),
+                    child: new Container(
+                      width: 100,
+                      height: 38,
+                      decoration: new BoxDecoration(
+                        color: Color(0xFF2ECC71),
+                        borderRadius: new BorderRadius.circular(5),
+                      ),
+                      child: new Center(
+                        child: new Text(
+                          'RUN',
+                          style: new TextStyle(fontSize: 14.0, fontWeight: FontWeight.bold, color: Colors.white),
+                        ),
+                      ),
+                    ),
+                  ),
+                ),
+                Expanded(
+                  child: Container(
+                      alignment: Alignment(-1.0, -1.0),
+                      margin: EdgeInsets.all(20.0),
+                      padding: EdgeInsets.all(4.0),
+                      decoration: new BoxDecoration(borderRadius: BorderRadius.all(new Radius.circular(5)), color: Color(0xFFF1C40F)),
+                      child: SingleChildScrollView(reverse: true, child: Text(_commandOutput))),
+                )
+              ],
+            ),
+            Column(
+              crossAxisAlignment: CrossAxisAlignment.center,
+              children: <Widget>[
+                Container(
+                    padding: const EdgeInsets.fromLTRB(20, 40, 20, 40),
+                    child: Container(
+                      width: 200,
+                      alignment: Alignment(0.0, 0.0),
+                      decoration: BoxDecoration(color: Color.fromRGBO(155, 89, 182, 1.0), borderRadius: BorderRadius.circular(5)),
+                      child: DropdownButtonHideUnderline(
+                          child: DropdownButton(
+                        style: new TextStyle(
+                          fontSize: 14,
+                          color: Colors.black,
+                        ),
+                        value: _currentCodec,
+                        items: _codecDropDownMenuItems,
+                        onChanged: _changedCodec,
+                        iconSize: 0,
+                        isExpanded: false,
+                      )),
+                    )),
+                Container(
+                  padding: const EdgeInsets.only(bottom: 20),
+                  child: new InkWell(
+                    onTap: () => testEncodeVideo(),
+                    child: new Container(
+                      width: 100,
+                      height: 38,
+                      decoration: new BoxDecoration(
+                        color: Color(0xFF2ECC71),
+                        borderRadius: new BorderRadius.circular(5),
+                      ),
+                      child: new Center(
+                        child: new Text(
+                          'ENCODE',
+                          style: new TextStyle(fontSize: 14.0, fontWeight: FontWeight.bold, color: Colors.white),
+                        ),
+                      ),
+                    ),
+                  ),
+                ),
+                Expanded(
+                  child: Container(
+                      alignment: Alignment(-1.0, -1.0),
+                      margin: EdgeInsets.all(20.0),
+                      padding: EdgeInsets.all(4.0),
+                      decoration: new BoxDecoration(borderRadius: BorderRadius.all(new Radius.circular(5)), color: Color(0xFFF1C40F)),
+                      child: SingleChildScrollView(reverse: true, child: Text(_encodeOutput))),
+                )
+              ],
+            ),
+          ],
+          controller: _controller,
+        ));
+  }
+
+  @override
+  void dispose() {
+    super.dispose();
+    _commandController.dispose();
+  }
+}

+ 5 - 0
example/lib/main.dart

@@ -0,0 +1,5 @@
+import 'package:flutter/material.dart';
+
+import 'flutter_ffmpeg_test_app.dart';
+
+void main() => runApp(FlutterFFmpegTestApp());

+ 46 - 0
example/lib/video_util.dart

@@ -0,0 +1,46 @@
+import 'dart:io';
+
+import 'package:flutter/services.dart';
+import 'package:path/path.dart';
+import 'package:path_provider/path_provider.dart';
+
+class VideoUtil {
+  static Future<Directory> get tempDirectory async {
+    return await getTemporaryDirectory();
+  }
+
+  static Future<File> copyFileAssets(String assetName, String localName) async {
+    final ByteData assetByteData = await rootBundle.load(assetName);
+
+    final List<int> byteList = assetByteData.buffer.asUint8List(assetByteData.offsetInBytes, assetByteData.lengthInBytes);
+
+    final String fullTemporaryPath = join((await tempDirectory).path, localName);
+
+    return new File(fullTemporaryPath).writeAsBytes(byteList, mode: FileMode.writeOnly, flush: true);
+  }
+
+  static Future<String> assetPath(String assetName) async {
+    return join((await tempDirectory).path, assetName);
+  }
+
+  static String generateEncodeVideoScript(String image1Path, String image2Path, String image3Path, String videoFilePath, String videoCodec, String customOptions) {
+    return "-hide_banner -y -loop 1 -i " + image1Path + " " +
+        "-loop 1 -i " + image2Path + " " +
+        "-loop 1 -i " + image3Path + " " +
+        "-filter_complex " +
+        "[0:v]setpts=PTS-STARTPTS,scale=w=\'if(gte(iw/ih,640/427),min(iw,640),-1)\':h=\'if(gte(iw/ih,640/427),-1,min(ih,427))\',scale=trunc(iw/2)*2:trunc(ih/2)*2,setsar=sar=1/1,split=2[stream1out1][stream1out2];" +
+        "[1:v]setpts=PTS-STARTPTS,scale=w=\'if(gte(iw/ih,640/427),min(iw,640),-1)\':h=\'if(gte(iw/ih,640/427),-1,min(ih,427))\',scale=trunc(iw/2)*2:trunc(ih/2)*2,setsar=sar=1/1,split=2[stream2out1][stream2out2];" +
+        "[2:v]setpts=PTS-STARTPTS,scale=w=\'if(gte(iw/ih,640/427),min(iw,640),-1)\':h=\'if(gte(iw/ih,640/427),-1,min(ih,427))\',scale=trunc(iw/2)*2:trunc(ih/2)*2,setsar=sar=1/1,split=2[stream3out1][stream3out2];" +
+        "[stream1out1]pad=width=640:height=427:x=(640-iw)/2:y=(427-ih)/2:color=#00000000,trim=duration=3,select=lte(n\\,90)[stream1overlaid];" +
+        "[stream1out2]pad=width=640:height=427:x=(640-iw)/2:y=(427-ih)/2:color=#00000000,trim=duration=1,select=lte(n\\,30)[stream1ending];" +
+        "[stream2out1]pad=width=640:height=427:x=(640-iw)/2:y=(427-ih)/2:color=#00000000,trim=duration=2,select=lte(n\\,60)[stream2overlaid];" +
+        "[stream2out2]pad=width=640:height=427:x=(640-iw)/2:y=(427-ih)/2:color=#00000000,trim=duration=1,select=lte(n\\,30),split=2[stream2starting][stream2ending];" +
+        "[stream3out1]pad=width=640:height=427:x=(640-iw)/2:y=(427-ih)/2:color=#00000000,trim=duration=2,select=lte(n\\,60)[stream3overlaid];" +
+        "[stream3out2]pad=width=640:height=427:x=(640-iw)/2:y=(427-ih)/2:color=#00000000,trim=duration=1,select=lte(n\\,30)[stream3starting];" +
+        "[stream2starting][stream1ending]blend=all_expr=\'if(gte(X,(W/2)*T/1)*lte(X,W-(W/2)*T/1),B,A)\':shortest=1[stream2blended];" +
+        "[stream3starting][stream2ending]blend=all_expr=\'if(gte(X,(W/2)*T/1)*lte(X,W-(W/2)*T/1),B,A)\':shortest=1[stream3blended];" +
+        "[stream1overlaid][stream2blended][stream2overlaid][stream3blended][stream3overlaid]concat=n=5:v=1:a=0,scale=w=640:h=424,format=yuv420p[video]" +
+        " -map [video] -vsync 2 -async 1 " + customOptions + "-c:v " + videoCodec + " -r 30 " + videoFilePath;
+  }
+
+}

+ 23 - 0
example/pubspec.yaml

@@ -0,0 +1,23 @@
+name: flutter_ffmpeg_example
+description: Demonstrates how to use the flutter_ffmpeg plugin.
+publish_to: 'none'
+
+environment:
+  sdk: ">=2.0.0-dev.68.0 <3.0.0"
+
+dependencies:
+  flutter:
+    sdk: flutter
+  cupertino_icons: ^0.1.2
+  path_provider:
+  flutter_ffmpeg:
+    path: ../
+
+dev_dependencies:
+  flutter_test:
+    sdk: flutter
+
+flutter:
+  uses-material-design: true
+  assets:
+    - assets/

+ 26 - 0
example/test/widget_test.dart

@@ -0,0 +1,26 @@
+// This is a basic Flutter widget test.
+//
+// To perform an interaction with a widget in your test, use the WidgetTester
+// utility that Flutter provides. For example, you can send tap and scroll
+// gestures. You can also use WidgetTester to find child widgets in the widget
+// tree, read text, and verify that the values of widget properties are correct.
+
+import 'package:flutter/material.dart';
+import 'package:flutter_test/flutter_test.dart';
+
+import 'package:flutter_ffmpeg_example/flutter_ffmpeg_test_app.dart';
+
+void main() {
+  testWidgets('Verify Platform version', (WidgetTester tester) async {
+    // Build our app and trigger a frame.
+    await tester.pumpWidget(FlutterFFmpegTestApp());
+
+    // Verify that platform version is retrieved.
+    expect(
+      find.byWidgetPredicate(
+        (Widget widget) => widget is Text && widget.data.startsWith('Running on:'),
+      ),
+      findsOneWidget,
+    );
+  });
+}

+ 20 - 0
flutter_ffmpeg.iml

@@ -0,0 +1,20 @@
+<?xml version="1.0" encoding="UTF-8"?>
+<module type="JAVA_MODULE" version="4">
+  <component name="NewModuleRootManager" inherit-compiler-output="true">
+    <exclude-output />
+    <content url="file://$MODULE_DIR$">
+      <sourceFolder url="file://$MODULE_DIR$/lib" isTestSource="false" />
+      <excludeFolder url="file://$MODULE_DIR$/.dart_tool" />
+      <excludeFolder url="file://$MODULE_DIR$/.idea" />
+      <excludeFolder url="file://$MODULE_DIR$/.pub" />
+      <excludeFolder url="file://$MODULE_DIR$/build" />
+      <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/Flutter/flutter_assets/packages" />
+    </content>
+    <orderEntry type="sourceFolder" forTests="false" />
+    <orderEntry type="library" name="Dart SDK" level="project" />
+    <orderEntry type="library" name="Flutter Plugins" level="project" />
+  </component>
+</module>

+ 36 - 0
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
ios/Assets/.gitkeep


+ 24 - 0
ios/Classes/FlutterFFmpegPlugin.h

@@ -0,0 +1,24 @@
+/*
+ * 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>
+
+@interface FlutterFFmpegPlugin : NSObject<FlutterPlugin,FlutterStreamHandler,LogDelegate,StatisticsDelegate>
+@end

+ 384 - 0
ios/Classes/FlutterFfmpegPlugin.m

@@ -0,0 +1,384 @@
+/*
+ * 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";
+
+@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
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.LGPLv3.txt' }
+
+  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', '4.2.LTS'
+
+end
+

+ 352 - 0
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
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
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