[
  {
    "path": ".gitignore",
    "content": ".gradle\nlocal.properties\n.idea\n.DS_Store\nbuild/\nreports/\ngradle.properties\n*.iml"
  },
  {
    "path": "CHANGES.md",
    "content": "# The revision history of android-ribbonizer-plugin\n\n## v2.1.0 2017/08/19\n\n* Allow null values for filters and ribbon labels\n* Added `largeRibbon` option to increase visibility in launcher icon foreground assets\n\n## v2.0.0 2017/07/21\n\nNo code change from v1.1.0; the reason of bumping the version is that v1.1.0 has breaking changes in the artifact id (`s/plugin/ribbonizer-plugin/`).\n\n## v1.1.0 2017/07/21\n\nhttps://github.com/gfx/gradle-android-ribbonizer-plugin/compare/v1.0.0...v1.1.0\n\n* Change the artifact id from `plugin` to `ribbonizer-plugin`\n* [#16](https://github.com/gfx/gradle-android-ribbonizer-plugin/pull/16) Support for android:roundIcon launcher images\n* [#19](https://github.com/gfx/gradle-android-ribbonizer-plugin/pull/19) Exclude XML resources\n\n## v1.0.0 2016/11/16\n\n* Add `customColorRibbonFilter(variant, iconFile, color)` (#13)\n\n## v0.6.0 - 2016-06-16\n\n* Add `forcedVariantsNames` (#11)\n\n## v0.5.0 - 2015-11-19\n\n* Build with Android Gradle plugin v1.5.0\n\n## v0.4.0 - 2015-06-05\n\n* [Add RibbonizerExtension#grayScaleFilter by kazy1991 · Pull Request #4 · gfx/gradle-android-ribbonizer-plugin](https://github.com/gfx/gradle-android-ribbonizer-plugin/pull/4)\n* Add ways to add icon names (see `example-custom/build.gradle` for example)\n\n## v0.3.0 - 2015-04-15\n\n* Fix \"AndroidManifest.xml not found\" errors\n\n## v0.2.0 - 2015-04-12\n\n* Add `example-custom` to show how to use the `ribbonizer` extension block\n\n## v0.1.1 - 2015-04-03\n\n* Fix a crash caused by an Android Studio issue\n\n## v0.1.0 - 2015-03-29\n\n* Initial Release\n"
  },
  {
    "path": "LICENSE",
    "content": "The MIT License (MIT)\n\nCopyright (c) 2015 FUJI Goro (gfx) <gfuji@cpan.org>.\n\nPermission is hereby granted, free of charge, to any person obtaining a copy\nof this software and associated documentation files (the \"Software\"), to deal\nin the Software without restriction, including without limitation the rights\nto use, copy, modify, merge, publish, distribute, sublicense, and/or sell\ncopies of the Software, and to permit persons to whom the Software is\nfurnished to do so, subject to the following conditions:\n\nThe above copyright notice and this permission notice shall be included in\nall copies or substantial portions of the Software.\n\nTHE SOFTWARE IS PROVIDED \"AS IS\", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR\nIMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,\nFITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE\nAUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER\nLIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,\nOUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN\nTHE SOFTWARE."
  },
  {
    "path": "Makefile",
    "content": "check:\n\t./gradlew clean check bintrayUpload\n\npublish: check\n\t./gradlew releng\n\t./gradlew -PdryRun=false --info plugin:bintrayUpload\n\nupdate-examples:\n\t./gradlew ribbonize\n\tcp ./example-simple/build/generated/ribbonizer/res/debug/mipmap-xxhdpi/ic_launcher.png ic-debug.png\n\tcp ./example-custom/build/generated/ribbonizer/res/localBeta/mipmap-xxhdpi/ic_launcher.png ic-beta.png\n"
  },
  {
    "path": "README.md",
    "content": "# DEPRECATED\nNo longer actively maintained, I recommend [Mikel's easylauncher-gradle-plugin](https://github.com/akaita/easylauncher-gradle-plugin) which might not have seen much love lately either, but seems to still work since the JFrog shutdown.\n\n# Ribbonizer plugin for Android\n\n[![Circle CI](https://circleci.com/gh/gfx/gradle-android-ribbonizer-plugin.svg?style=svg&branch=master)](https://circleci.com/gh/gfx/gradle-android-ribbonizer-plugin) [![Download](https://api.bintray.com/packages/gfx/maven/ribbonizer-plugin/images/download.svg)](https://bintray.com/gfx/maven/ribbonizer-plugin/_latestVersion)\n\nThis is a ribbonizer as a Gradle plugin for Android, which adds a ribbon to launcher icons.\n\n![](ic-beta.png) ![](ic-debug.png)\n\n## Usage\n\n```groovy\n// in build.gradle\nbuildscript {\n    repositories {\n        jcenter()\n    }\n    dependencies {\n        classpath 'com.android.tools.build:gradle:2.1.2'\n        classpath 'com.github.gfx.ribbonizer:ribbonizer-plugin:2.1.0'\n    }\n}\n```\n\n```groovy\n// in app/build.gradle\napply plugin: 'com.github.gfx.ribbonizer'\n\nandroid {\n    // ...\n\n    buildTypes {\n        debug {/*debuggable build, which will ribbonized automatically*/}\n        beta {\n            //debuggable build which will automatically ribbonized.\n            debuggable true\n        }\n        canary {\n            //non-debuggable build which will no automatically ribbonized.\n            //But, we force one of its flavors. See `ribbonizer` for how-to\n            debuggable false\n        }\n        release {/*non-debuggable build. Will not be rebbonized automatically*/}\n    }\n\n    productFlavors {\n        local {}\n        qa {}\n        staging {}\n        production {}\n    }\n}\n\nribbonizer {\n    // \"manifest application[android:icon]\" is automatically added to the list\n    iconNames \"@drawable/ic_notification\", \"@drawable/widget_preview\"\n\n    builder { variant, iconFile ->\n        // change ribbon colors by product flavors\n        if (variant.flavorName == \"local\") {\n            return grayRibbonFilter(variant, iconFile)\n        } else if (variant.flavorName == \"qa\") {\n            // customColorRibbonFilter allows setting any color code\n            def filter = customColorRibbonFilter(variant, iconFile, \"#00C89C\")\n            // Finer control of the label text can be achieved by setting it manually, or set to\n            // null for an unlabelled ribbon. The default is to use the flavor name.\n            filter.label = \"QA\" + variant.versionCode\n            return filter\n        } else if (variant.flavorName == \"staging\") {\n            return yellowRibbonFilter(variant, iconFile)\n        } else if (variant.buildType.name == \"debug\") {\n            if (variant.flavorName == \"production\") {\n                // Particular configurations can be skipped by returning no filters\n                return null\n            }\n            else {\n                // Other filters can be applied, as long as they implement Consumer<BufferedImage>\n                return grayScaleFilter(variant, iconFile)\n            }\n        } else {\n            return greenRibbonFilter(variant, iconFile)\n        }\n    }\n\n    //Although `canary` build-type is marked as `non-debuggable`\n    //we can still force specific variants to be ribbonized:\n    forcedVariantsNames \"localCanary\"\n}\n\n```\n\n\n## Project Structure\n\n```\nplugin/   - The main module of a Gradle plugin\nexample/  - An example android application that uses this plugin\nbuildSrc/ - A helper module to use this plugin in example modules\n```\n\nYou can test this project with `./gradlew check`.\n\n## Release Engineering\n\n```console\n./gradlew bumpMinor # or bumpMajor, bumpPatch\n\nmake publish # upload artifacts to bintray jcenter\n```\n\n## Author And License\n\nThe MIT License (MIT)\n\nCopyright (c) 2015 FUJI Goro (gfx) <gfuji@cpan.org>.\n\nPermission is hereby granted, free of charge, to any person obtaining a copy\nof this software and associated documentation files (the \"Software\"), to deal\nin the Software without restriction, including without limitation the rights\nto use, copy, modify, merge, publish, distribute, sublicense, and/or sell\ncopies of the Software, and to permit persons to whom the Software is\nfurnished to do so, subject to the following conditions:\n\nThe above copyright notice and this permission notice shall be included in\nall copies or substantial portions of the Software.\n\nTHE SOFTWARE IS PROVIDED \"AS IS\", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR\nIMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,\nFITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE\nAUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER\nLIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,\nOUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN\nTHE SOFTWARE.\n"
  },
  {
    "path": "VERSION",
    "content": "2.1.0\n"
  },
  {
    "path": "build.gradle",
    "content": "apply from: './versioning.gradle'\napply from: './metadata.gradle'\n\next {\n    metadata.version = versionName\n}\n\nbuildscript {\n    repositories {\n        jcenter()\n    }\n    dependencies {\n        classpath 'com.android.tools.build:gradle:2.3.3'\n        classpath 'com.novoda:bintray-release:0.5.0' // https://github.com/novoda/bintray-release\n    }\n}\n\nallprojects {\n    repositories {\n        jcenter()\n    }\n}\n"
  },
  {
    "path": "buildSrc/build.gradle",
    "content": "System.setProperty(\"java.awt.headless\", \"true\")\n\nrepositories {\n    jcenter()\n}\n\ndependencies {\n    compile project(':plugin')\n}\n"
  },
  {
    "path": "buildSrc/settings.gradle",
    "content": "include ':plugin'\nproject(':plugin').projectDir = new File('../plugin')\n"
  },
  {
    "path": "circle.yml",
    "content": "machine:\n  java:\n    version: oraclejdk8\n  environment:\n    TERM: dumb\n    GRADLE_OPTS: '-Dorg.gradle.jvmargs=\"-Xmx1024m -XX:+HeapDumpOnOutOfMemoryError\" -Dorg.gradle.daemon=false'\ndependencies:\n  pre:\n    - echo y | android -s update sdk -u -a -t \"tools\" # update Android SDK that includes sdkmanager(1)\n    - mkdir -p \"$ANDROID_HOME\"/licenses\n    - echo \"8933bad161af4178b1185d1a37fbf41ea5269c55\" > \"$ANDROID_HOME\"/licenses/android-sdk-license\n    - $ANDROID_HOME/tools/bin/sdkmanager \"platform-tools\" \"extras;android;m2repository\"\ntest:\n  override:\n    - TERM=dumb ./gradlew check\n    - TERM=dumb ./gradlew ribbonize\n"
  },
  {
    "path": "example-custom/build.gradle",
    "content": "import com.android.build.gradle.api.ApplicationVariant\n\napply plugin: 'com.android.application'\napply plugin: 'com.github.gfx.ribbonizer'\n\nandroid {\n    compileSdkVersion 26\n    buildToolsVersion '26.0.0'\n\n    defaultConfig {\n        minSdkVersion 15\n        targetSdkVersion 26\n        versionCode 1\n        versionName \"1.0\"\n    }\n\n    compileOptions {\n        sourceCompatibility JavaVersion.VERSION_1_7\n        targetCompatibility JavaVersion.VERSION_1_7\n    }\n    lintOptions {\n        abortOnError false\n    }\n\n    buildTypes {\n        debug {}\n        beta {\n            debuggable true\n        }\n        canary {\n            debuggable false\n        }\n        release {}\n    }\n\n    productFlavors {\n        local {}\n        qa {}\n        staging {}\n        production {}\n    }\n}\n\nribbonizer {\n    iconNames \"@drawable/abc_btn_check_to_on_mtrl_000\", \"@drawable/abc_btn_radio_to_on_mtrl_000\", \"@mipmap/ic_launcher_foreground\"\n\n    builder { ApplicationVariant variant, File iconFile ->\n        if (variant.flavorName == \"local\") {\n            return grayRibbonFilter(variant, iconFile)\n        } else if (variant.flavorName == \"qa\") {\n            def filter = customColorRibbonFilter(variant, iconFile, \"#00C89C\")\n            filter.label = null//\"QA\" + variant.versionCode\n            filter.largeRibbon = (iconFile.name == \"ic_launcher_foreground.png\")\n            return filter\n        } else if (variant.buildType.name == \"debug\") {\n            if (variant.flavorName == \"production\") {\n                return null\n            }\n            return customColorRibbonFilter(variant, iconFile, \"#0000FF\")\n        } else {\n            return greenRibbonFilter(variant, iconFile)\n        }\n    }\n\n    forcedVariantsNames \"localCanary\"\n}\n\ndependencies {\n    compile 'com.jakewharton:butterknife:6.1.0'\n    compile 'com.android.support:appcompat-v7:25.0.0'\n}\n"
  },
  {
    "path": "example-custom/src/main/AndroidManifest.xml",
    "content": "<?xml version=\"1.0\" encoding=\"utf-8\"?>\n<manifest xmlns:android=\"http://schemas.android.com/apk/res/android\"\n    package=\"com.github.gfx.ribbonizer.example\">\n\n    <application\n        android:allowBackup=\"true\"\n        android:icon=\"@mipmap/ic_launcher\"\n        android:roundIcon=\"@mipmap/ic_launcher_round\"\n        android:label=\"@string/app_name\"\n        android:theme=\"@style/AppTheme\" >\n        <activity\n            android:name=\"com.github.gfx.ribbonizer.example.MainActivity\"\n            android:label=\"@string/app_name\" >\n            <intent-filter>\n                <action android:name=\"android.intent.action.MAIN\" />\n\n                <category android:name=\"android.intent.category.LAUNCHER\" />\n            </intent-filter>\n        </activity>\n    </application>\n\n</manifest>\n"
  },
  {
    "path": "example-custom/src/main/java/com/github/gfx/ribbonizer/example/MainActivity.java",
    "content": "package com.github.gfx.ribbonizer.example;\n\nimport android.app.Activity;\nimport android.app.AlertDialog;\nimport android.os.Bundle;\nimport android.widget.TextView;\n\nimport butterknife.ButterKnife;\nimport butterknife.InjectView;\nimport butterknife.OnClick;\n\npublic class MainActivity extends Activity {\n\n    @InjectView(R.id.text)\n    TextView textView;\n\n    @Override\n    protected void onCreate(Bundle savedInstanceState) {\n        super.onCreate(savedInstanceState);\n        setContentView(R.layout.activity_main);\n\n        ButterKnife.inject(this);\n    }\n\n    @OnClick(R.id.button)\n    void onButtonClick() {\n        new AlertDialog.Builder(this)\n                .setTitle(\"Hello, Android!\")\n                .setMessage(\"This is an example app.\")\n                .setPositiveButton(\"OK\", null)\n                .show();\n    }\n}\n"
  },
  {
    "path": "example-custom/src/main/res/layout/activity_main.xml",
    "content": "<LinearLayout xmlns:android=\"http://schemas.android.com/apk/res/android\"\n    xmlns:tools=\"http://schemas.android.com/tools\"\n    android:layout_width=\"match_parent\"\n    android:layout_height=\"match_parent\"\n    android:paddingLeft=\"@dimen/activity_horizontal_margin\"\n    android:paddingRight=\"@dimen/activity_horizontal_margin\"\n    android:paddingTop=\"@dimen/activity_vertical_margin\"\n    android:paddingBottom=\"@dimen/activity_vertical_margin\"\n    android:orientation=\"vertical\"\n    tools:context=\"com.github.gfx.ribbonizer.example.MainActivity\">\n\n    <TextView\n            android:id=\"@+id/text\"\n        android:layout_gravity=\"center_horizontal\"\n        android:textAppearance=\"@android:style/TextAppearance.DeviceDefault.Large\"\n        android:text=\"@string/hello_world\"\n        android:layout_width=\"wrap_content\"\n        android:layout_height=\"wrap_content\" />\n\n    <Button\n        android:layout_margin=\"22dp\"\n        android:layout_width=\"wrap_content\"\n        android:layout_height=\"wrap_content\"\n        android:text=\"@string/button\"\n        android:id=\"@+id/button\"\n        android:layout_gravity=\"center_horizontal\"/>\n\n</LinearLayout>\n"
  },
  {
    "path": "example-custom/src/main/res/mipmap-anydpi-v26/ic_launcher.xml",
    "content": "<?xml version=\"1.0\" encoding=\"utf-8\"?>\n<adaptive-icon xmlns:android=\"http://schemas.android.com/apk/res/android\">\n    <background android:drawable=\"@color/ic_launcher_background\"/>\n    <foreground android:drawable=\"@mipmap/ic_launcher_foreground\"/>\n</adaptive-icon>"
  },
  {
    "path": "example-custom/src/main/res/mipmap-anydpi-v26/ic_launcher_round.xml",
    "content": "<?xml version=\"1.0\" encoding=\"utf-8\"?>\n<adaptive-icon xmlns:android=\"http://schemas.android.com/apk/res/android\">\n    <background android:drawable=\"@color/ic_launcher_background\"/>\n    <foreground android:drawable=\"@mipmap/ic_launcher_foreground\"/>\n</adaptive-icon>"
  },
  {
    "path": "example-custom/src/main/res/values/dimens.xml",
    "content": "<resources>\n    <!-- Default screen margins, per the Android Design guidelines. -->\n    <dimen name=\"activity_horizontal_margin\">16dp</dimen>\n    <dimen name=\"activity_vertical_margin\">16dp</dimen>\n</resources>\n"
  },
  {
    "path": "example-custom/src/main/res/values/ic_launcher_background.xml",
    "content": "<?xml version=\"1.0\" encoding=\"utf-8\"?>\n<resources>\n  <color name=\"ic_launcher_background\">#00C89C</color>\n</resources>"
  },
  {
    "path": "example-custom/src/main/res/values/strings.xml",
    "content": "<?xml version=\"1.0\" encoding=\"utf-8\"?>\n<resources>\n\n    <string name=\"app_name\">App Icon Demo</string>\n    <string name=\"hello_world\">Hello, world!</string>\n    <string name=\"action_settings\">Settings</string>\n    <string name=\"button\">button</string>\n\n</resources>\n"
  },
  {
    "path": "example-custom/src/main/res/values/styles.xml",
    "content": "<resources>\n\n    <!-- Base application theme. -->\n    <style name=\"AppTheme\" parent=\"android:Theme.Holo.Light\">\n        <!-- Customize your theme here. -->\n    </style>\n\n</resources>\n"
  },
  {
    "path": "example-custom/src/main/res/values-v26/colors.xml",
    "content": "<?xml version=\"1.0\" encoding=\"utf-8\"?>\n<resources>\n    <color name=\"ic_launcher_background\">#1c7917</color>\n</resources>\n"
  },
  {
    "path": "example-custom/src/main/res/values-w820dp/dimens.xml",
    "content": "<resources>\n    <!-- Example customization of dimensions originally defined in res/values/dimens.xml\n         (such as screen margins) for screens with more than 820dp of available width. This\n         would include 7\" and 10\" devices in landscape (~960dp and ~1280dp respectively). -->\n    <dimen name=\"activity_horizontal_margin\">64dp</dimen>\n</resources>\n"
  },
  {
    "path": "example-simple/build.gradle",
    "content": "apply plugin: 'com.android.application'\napply plugin: 'com.github.gfx.ribbonizer'\n\nandroid {\n    compileSdkVersion 25\n    buildToolsVersion '25'\n\n    defaultConfig {\n        minSdkVersion 15\n        targetSdkVersion 23\n        versionCode 1\n        versionName \"1.0\"\n    }\n\n    compileOptions {\n        sourceCompatibility JavaVersion.VERSION_1_7\n        targetCompatibility JavaVersion.VERSION_1_7\n    }\n    lintOptions {\n        abortOnError false\n    }\n}\n\n\ndependencies {\n    compile 'com.jakewharton:butterknife:6.1.0'\n}\n"
  },
  {
    "path": "example-simple/src/main/AndroidManifest.xml",
    "content": "<?xml version=\"1.0\" encoding=\"utf-8\"?>\n<manifest xmlns:android=\"http://schemas.android.com/apk/res/android\"\n    package=\"com.github.gfx.ribbonizer.example\">\n\n    <application\n        android:allowBackup=\"true\"\n        android:icon=\"@drawable/ic_launcher\"\n        android:label=\"@string/app_name\"\n        android:theme=\"@style/AppTheme\" >\n        <activity\n            android:name=\"com.github.gfx.ribbonizer.example.MainActivity\"\n            android:label=\"@string/app_name\" >\n            <intent-filter>\n                <action android:name=\"android.intent.action.MAIN\" />\n\n                <category android:name=\"android.intent.category.LAUNCHER\" />\n            </intent-filter>\n        </activity>\n    </application>\n\n</manifest>\n"
  },
  {
    "path": "example-simple/src/main/java/com/github/gfx/ribbonizer/example/MainActivity.java",
    "content": "package com.github.gfx.ribbonizer.example;\n\nimport android.app.Activity;\nimport android.app.AlertDialog;\nimport android.os.Bundle;\nimport android.widget.TextView;\n\nimport butterknife.ButterKnife;\nimport butterknife.InjectView;\nimport butterknife.OnClick;\n\npublic class MainActivity extends Activity {\n\n    @InjectView(R.id.text)\n    TextView textView;\n\n    @Override\n    protected void onCreate(Bundle savedInstanceState) {\n        super.onCreate(savedInstanceState);\n        setContentView(R.layout.activity_main);\n\n        ButterKnife.inject(this);\n    }\n\n    @OnClick(R.id.button)\n    void onButtonClick() {\n        new AlertDialog.Builder(this)\n                .setTitle(\"Hello, Android!\")\n                .setMessage(\"This is an example app.\")\n                .setPositiveButton(\"OK\", null)\n                .show();\n    }\n}\n"
  },
  {
    "path": "example-simple/src/main/res/layout/activity_main.xml",
    "content": "<LinearLayout xmlns:android=\"http://schemas.android.com/apk/res/android\"\n    xmlns:tools=\"http://schemas.android.com/tools\"\n    android:layout_width=\"match_parent\"\n    android:layout_height=\"match_parent\"\n    android:paddingLeft=\"@dimen/activity_horizontal_margin\"\n    android:paddingRight=\"@dimen/activity_horizontal_margin\"\n    android:paddingTop=\"@dimen/activity_vertical_margin\"\n    android:paddingBottom=\"@dimen/activity_vertical_margin\"\n    android:orientation=\"vertical\"\n    tools:context=\"com.github.gfx.ribbonizer.example.MainActivity\">\n\n    <TextView\n            android:id=\"@+id/text\"\n        android:layout_gravity=\"center_horizontal\"\n        android:textAppearance=\"@android:style/TextAppearance.DeviceDefault.Large\"\n        android:text=\"@string/hello_world\"\n        android:layout_width=\"wrap_content\"\n        android:layout_height=\"wrap_content\" />\n\n    <Button\n        android:layout_margin=\"22dp\"\n        android:layout_width=\"wrap_content\"\n        android:layout_height=\"wrap_content\"\n        android:text=\"@string/button\"\n        android:id=\"@+id/button\"\n        android:layout_gravity=\"center_horizontal\"/>\n\n</LinearLayout>\n"
  },
  {
    "path": "example-simple/src/main/res/values/dimens.xml",
    "content": "<resources>\n    <!-- Default screen margins, per the Android Design guidelines. -->\n    <dimen name=\"activity_horizontal_margin\">16dp</dimen>\n    <dimen name=\"activity_vertical_margin\">16dp</dimen>\n</resources>\n"
  },
  {
    "path": "example-simple/src/main/res/values/strings.xml",
    "content": "<?xml version=\"1.0\" encoding=\"utf-8\"?>\n<resources>\n\n    <string name=\"app_name\">App Icon Demo</string>\n    <string name=\"hello_world\">Hello, world!</string>\n    <string name=\"action_settings\">Settings</string>\n    <string name=\"button\">button</string>\n\n</resources>\n"
  },
  {
    "path": "example-simple/src/main/res/values/styles.xml",
    "content": "<resources>\n\n    <!-- Base application theme. -->\n    <style name=\"AppTheme\" parent=\"android:Theme.Holo.Light\">\n        <!-- Customize your theme here. -->\n    </style>\n\n</resources>\n"
  },
  {
    "path": "example-simple/src/main/res/values-w820dp/dimens.xml",
    "content": "<resources>\n    <!-- Example customization of dimensions originally defined in res/values/dimens.xml\n         (such as screen margins) for screens with more than 820dp of available width. This\n         would include 7\" and 10\" devices in landscape (~960dp and ~1280dp respectively). -->\n    <dimen name=\"activity_horizontal_margin\">64dp</dimen>\n</resources>\n"
  },
  {
    "path": "gradle/wrapper/gradle-wrapper.properties",
    "content": "#Thu Dec 11 08:17:04 JST 2014\ndistributionBase=GRADLE_USER_HOME\ndistributionPath=wrapper/dists\nzipStoreBase=GRADLE_USER_HOME\nzipStorePath=wrapper/dists\ndistributionUrl=https\\://services.gradle.org/distributions/gradle-3.5-all.zip\n"
  },
  {
    "path": "gradle.properties.sample",
    "content": "# configuration for bintray and signing\n\nbintrayUser=YOUR_USERNAME\nbintrayKey=YOUR_KEY\n"
  },
  {
    "path": "gradlew",
    "content": "#!/usr/bin/env bash\n\n##############################################################################\n##\n##  Gradle start up script for UN*X\n##\n##############################################################################\n\n# Add default JVM options here. You can also use JAVA_OPTS and GRADLE_OPTS to pass JVM options to this script.\nDEFAULT_JVM_OPTS=\"\"\n\nAPP_NAME=\"Gradle\"\nAPP_BASE_NAME=`basename \"$0\"`\n\n# Use the maximum available, or set MAX_FD != -1 to use that value.\nMAX_FD=\"maximum\"\n\nwarn ( ) {\n    echo \"$*\"\n}\n\ndie ( ) {\n    echo\n    echo \"$*\"\n    echo\n    exit 1\n}\n\n# OS specific support (must be 'true' or 'false').\ncygwin=false\nmsys=false\ndarwin=false\ncase \"`uname`\" in\n  CYGWIN* )\n    cygwin=true\n    ;;\n  Darwin* )\n    darwin=true\n    ;;\n  MINGW* )\n    msys=true\n    ;;\nesac\n\n# For Cygwin, ensure paths are in UNIX format before anything is touched.\nif $cygwin ; then\n    [ -n \"$JAVA_HOME\" ] && JAVA_HOME=`cygpath --unix \"$JAVA_HOME\"`\nfi\n\n# Attempt to set APP_HOME\n# Resolve links: $0 may be a link\nPRG=\"$0\"\n# Need this for relative symlinks.\nwhile [ -h \"$PRG\" ] ; do\n    ls=`ls -ld \"$PRG\"`\n    link=`expr \"$ls\" : '.*-> \\(.*\\)$'`\n    if expr \"$link\" : '/.*' > /dev/null; then\n        PRG=\"$link\"\n    else\n        PRG=`dirname \"$PRG\"`\"/$link\"\n    fi\ndone\nSAVED=\"`pwd`\"\ncd \"`dirname \\\"$PRG\\\"`/\" >&-\nAPP_HOME=\"`pwd -P`\"\ncd \"$SAVED\" >&-\n\nCLASSPATH=$APP_HOME/gradle/wrapper/gradle-wrapper.jar\n\n# Determine the Java command to use to start the JVM.\nif [ -n \"$JAVA_HOME\" ] ; then\n    if [ -x \"$JAVA_HOME/jre/sh/java\" ] ; then\n        # IBM's JDK on AIX uses strange locations for the executables\n        JAVACMD=\"$JAVA_HOME/jre/sh/java\"\n    else\n        JAVACMD=\"$JAVA_HOME/bin/java\"\n    fi\n    if [ ! -x \"$JAVACMD\" ] ; then\n        die \"ERROR: JAVA_HOME is set to an invalid directory: $JAVA_HOME\n\nPlease set the JAVA_HOME variable in your environment to match the\nlocation of your Java installation.\"\n    fi\nelse\n    JAVACMD=\"java\"\n    which java >/dev/null 2>&1 || die \"ERROR: JAVA_HOME is not set and no 'java' command could be found in your PATH.\n\nPlease set the JAVA_HOME variable in your environment to match the\nlocation of your Java installation.\"\nfi\n\n# Increase the maximum file descriptors if we can.\nif [ \"$cygwin\" = \"false\" -a \"$darwin\" = \"false\" ] ; then\n    MAX_FD_LIMIT=`ulimit -H -n`\n    if [ $? -eq 0 ] ; then\n        if [ \"$MAX_FD\" = \"maximum\" -o \"$MAX_FD\" = \"max\" ] ; then\n            MAX_FD=\"$MAX_FD_LIMIT\"\n        fi\n        ulimit -n $MAX_FD\n        if [ $? -ne 0 ] ; then\n            warn \"Could not set maximum file descriptor limit: $MAX_FD\"\n        fi\n    else\n        warn \"Could not query maximum file descriptor limit: $MAX_FD_LIMIT\"\n    fi\nfi\n\n# For Darwin, add options to specify how the application appears in the dock\nif $darwin; then\n    GRADLE_OPTS=\"$GRADLE_OPTS \\\"-Xdock:name=$APP_NAME\\\" \\\"-Xdock:icon=$APP_HOME/media/gradle.icns\\\"\"\nfi\n\n# For Cygwin, switch paths to Windows format before running java\nif $cygwin ; then\n    APP_HOME=`cygpath --path --mixed \"$APP_HOME\"`\n    CLASSPATH=`cygpath --path --mixed \"$CLASSPATH\"`\n\n    # We build the pattern for arguments to be converted via cygpath\n    ROOTDIRSRAW=`find -L / -maxdepth 1 -mindepth 1 -type d 2>/dev/null`\n    SEP=\"\"\n    for dir in $ROOTDIRSRAW ; do\n        ROOTDIRS=\"$ROOTDIRS$SEP$dir\"\n        SEP=\"|\"\n    done\n    OURCYGPATTERN=\"(^($ROOTDIRS))\"\n    # Add a user-defined pattern to the cygpath arguments\n    if [ \"$GRADLE_CYGPATTERN\" != \"\" ] ; then\n        OURCYGPATTERN=\"$OURCYGPATTERN|($GRADLE_CYGPATTERN)\"\n    fi\n    # Now convert the arguments - kludge to limit ourselves to /bin/sh\n    i=0\n    for arg in \"$@\" ; do\n        CHECK=`echo \"$arg\"|egrep -c \"$OURCYGPATTERN\" -`\n        CHECK2=`echo \"$arg\"|egrep -c \"^-\"`                                 ### Determine if an option\n\n        if [ $CHECK -ne 0 ] && [ $CHECK2 -eq 0 ] ; then                    ### Added a condition\n            eval `echo args$i`=`cygpath --path --ignore --mixed \"$arg\"`\n        else\n            eval `echo args$i`=\"\\\"$arg\\\"\"\n        fi\n        i=$((i+1))\n    done\n    case $i in\n        (0) set -- ;;\n        (1) set -- \"$args0\" ;;\n        (2) set -- \"$args0\" \"$args1\" ;;\n        (3) set -- \"$args0\" \"$args1\" \"$args2\" ;;\n        (4) set -- \"$args0\" \"$args1\" \"$args2\" \"$args3\" ;;\n        (5) set -- \"$args0\" \"$args1\" \"$args2\" \"$args3\" \"$args4\" ;;\n        (6) set -- \"$args0\" \"$args1\" \"$args2\" \"$args3\" \"$args4\" \"$args5\" ;;\n        (7) set -- \"$args0\" \"$args1\" \"$args2\" \"$args3\" \"$args4\" \"$args5\" \"$args6\" ;;\n        (8) set -- \"$args0\" \"$args1\" \"$args2\" \"$args3\" \"$args4\" \"$args5\" \"$args6\" \"$args7\" ;;\n        (9) set -- \"$args0\" \"$args1\" \"$args2\" \"$args3\" \"$args4\" \"$args5\" \"$args6\" \"$args7\" \"$args8\" ;;\n    esac\nfi\n\n# Split up the JVM_OPTS And GRADLE_OPTS values into an array, following the shell quoting and substitution rules\nfunction splitJvmOpts() {\n    JVM_OPTS=(\"$@\")\n}\neval splitJvmOpts $DEFAULT_JVM_OPTS $JAVA_OPTS $GRADLE_OPTS\nJVM_OPTS[${#JVM_OPTS[*]}]=\"-Dorg.gradle.appname=$APP_BASE_NAME\"\n\nexec \"$JAVACMD\" \"${JVM_OPTS[@]}\" -classpath \"$CLASSPATH\" org.gradle.wrapper.GradleWrapperMain \"$@\"\n"
  },
  {
    "path": "gradlew.bat",
    "content": "@if \"%DEBUG%\" == \"\" @echo off\r\n@rem ##########################################################################\r\n@rem\r\n@rem  Gradle startup script for Windows\r\n@rem\r\n@rem ##########################################################################\r\n\r\n@rem Set local scope for the variables with windows NT shell\r\nif \"%OS%\"==\"Windows_NT\" setlocal\r\n\r\n@rem Add default JVM options here. You can also use JAVA_OPTS and GRADLE_OPTS to pass JVM options to this script.\r\nset DEFAULT_JVM_OPTS=\r\n\r\nset DIRNAME=%~dp0\r\nif \"%DIRNAME%\" == \"\" set DIRNAME=.\r\nset APP_BASE_NAME=%~n0\r\nset APP_HOME=%DIRNAME%\r\n\r\n@rem Find java.exe\r\nif defined JAVA_HOME goto findJavaFromJavaHome\r\n\r\nset JAVA_EXE=java.exe\r\n%JAVA_EXE% -version >NUL 2>&1\r\nif \"%ERRORLEVEL%\" == \"0\" goto init\r\n\r\necho.\r\necho ERROR: JAVA_HOME is not set and no 'java' command could be found in your PATH.\r\necho.\r\necho Please set the JAVA_HOME variable in your environment to match the\r\necho location of your Java installation.\r\n\r\ngoto fail\r\n\r\n:findJavaFromJavaHome\r\nset JAVA_HOME=%JAVA_HOME:\"=%\r\nset JAVA_EXE=%JAVA_HOME%/bin/java.exe\r\n\r\nif exist \"%JAVA_EXE%\" goto init\r\n\r\necho.\r\necho ERROR: JAVA_HOME is set to an invalid directory: %JAVA_HOME%\r\necho.\r\necho Please set the JAVA_HOME variable in your environment to match the\r\necho location of your Java installation.\r\n\r\ngoto fail\r\n\r\n:init\r\n@rem Get command-line arguments, handling Windowz variants\r\n\r\nif not \"%OS%\" == \"Windows_NT\" goto win9xME_args\r\nif \"%@eval[2+2]\" == \"4\" goto 4NT_args\r\n\r\n:win9xME_args\r\n@rem Slurp the command line arguments.\r\nset CMD_LINE_ARGS=\r\nset _SKIP=2\r\n\r\n:win9xME_args_slurp\r\nif \"x%~1\" == \"x\" goto execute\r\n\r\nset CMD_LINE_ARGS=%*\r\ngoto execute\r\n\r\n:4NT_args\r\n@rem Get arguments from the 4NT Shell from JP Software\r\nset CMD_LINE_ARGS=%$\r\n\r\n:execute\r\n@rem Setup the command line\r\n\r\nset CLASSPATH=%APP_HOME%\\gradle\\wrapper\\gradle-wrapper.jar\r\n\r\n@rem Execute Gradle\r\n\"%JAVA_EXE%\" %DEFAULT_JVM_OPTS% %JAVA_OPTS% %GRADLE_OPTS% \"-Dorg.gradle.appname=%APP_BASE_NAME%\" -classpath \"%CLASSPATH%\" org.gradle.wrapper.GradleWrapperMain %CMD_LINE_ARGS%\r\n\r\n:end\r\n@rem End local scope for the variables with windows NT shell\r\nif \"%ERRORLEVEL%\"==\"0\" goto mainEnd\r\n\r\n:fail\r\nrem Set variable GRADLE_EXIT_CONSOLE if you need the _script_ return code instead of\r\nrem the _cmd.exe /c_ return code!\r\nif  not \"\" == \"%GRADLE_EXIT_CONSOLE%\" exit 1\r\nexit /b 1\r\n\r\n:mainEnd\r\nif \"%OS%\"==\"Windows_NT\" endlocal\r\n\r\n:omega\r\n"
  },
  {
    "path": "metadata.gradle",
    "content": "ext {\n    metadata = [\n            userOrg   : 'ribbonizer',\n            groupId   : 'com.github.gfx.ribbonizer',\n            website   : 'https://github.com/maskarade/gradle-android-ribbonizer-plugin',\n            repository: 'https://github.com/maskarade/gradle-android-ribbonizer-plugin.git',\n            licences  : ['Apache-2.0']\n    ]\n}\n"
  },
  {
    "path": "plugin/build.gradle",
    "content": "repositories {\n    jcenter()\n}\n\napply plugin: 'groovy'\n\n\nsourceCompatibility = JavaVersion.VERSION_1_8\ntargetCompatibility = JavaVersion.VERSION_1_8\n\ndependencies {\n    compile gradleApi()\n    compile localGroovy()\n    compile 'com.android.tools.build:gradle:2.3.3'\n\n    testCompile('org.spockframework:spock-core:1.0-groovy-2.4') {\n        exclude group:'org.codehaus.groovy'\n    }\n}\n\nif (rootProject.hasProperty('versionName') && rootProject.hasProperty('bintrayUser')) {\n    jar {\n        manifest {\n            attributes(\n                    \"Implementation-Version\": project.file(\"../VERSION\").text.trim(),\n            )\n        }\n    }\n\n    apply plugin: 'com.novoda.bintray-release'\n\n    publish {\n        artifactId = 'ribbonizer-plugin'\n        desc = 'Gradle plugin to modifiy Android launcher icons'\n\n        def metadata = rootProject.ext.metadata\n        userOrg = metadata.userOrg\n        groupId = metadata.groupId\n        publishVersion = metadata.version\n        website = metadata.website\n        repository = metadata.repository\n        licences = metadata.licences\n    }\n}\n"
  },
  {
    "path": "plugin/src/main/groovy/com/github/gfx/ribbonizer/CustomColorRibbonBuilder.java",
    "content": "package com.github.gfx.ribbonizer;\n\nimport com.android.build.gradle.api.ApplicationVariant;\nimport com.github.gfx.ribbonizer.filter.ColorRibbonFilter;\n\nimport java.awt.Color;\nimport java.awt.image.BufferedImage;\nimport java.io.File;\nimport java.util.function.Consumer;\n\npublic class CustomColorRibbonBuilder implements FilterBuilder {\n\n    private String nm;\n\n    public CustomColorRibbonBuilder(String nm) {\n        this.nm = nm;\n    }\n\n    @Override\n    public Consumer<BufferedImage> apply(ApplicationVariant variant, File iconFile) {\n        return new ColorRibbonFilter(variant.getBuildType().getName(), Color.decode(nm));\n    }\n}\n"
  },
  {
    "path": "plugin/src/main/groovy/com/github/gfx/ribbonizer/FilterBuilder.java",
    "content": "package com.github.gfx.ribbonizer;\n\nimport com.android.build.gradle.api.ApplicationVariant;\n\nimport java.awt.image.BufferedImage;\nimport java.io.File;\nimport java.util.function.BiFunction;\nimport java.util.function.Consumer;\n\npublic interface FilterBuilder extends BiFunction<ApplicationVariant, File, Consumer<BufferedImage>> {\n\n}\n"
  },
  {
    "path": "plugin/src/main/groovy/com/github/gfx/ribbonizer/GrayRibbonBuilder.java",
    "content": "package com.github.gfx.ribbonizer;\n\nimport com.android.build.gradle.api.ApplicationVariant;\nimport com.github.gfx.ribbonizer.filter.ColorRibbonFilter;\n\nimport java.awt.*;\nimport java.awt.image.BufferedImage;\nimport java.io.File;\nimport java.util.function.Consumer;\n\npublic class GrayRibbonBuilder implements FilterBuilder {\n\n    public static final Color COLOR = new Color(0x60, 0x60, 0x60, 0x99);\n\n    @Override\n    public Consumer<BufferedImage> apply(ApplicationVariant variant, File iconFile) {\n        return new ColorRibbonFilter(variant.getBuildType().getName(), COLOR);\n    }\n}\n"
  },
  {
    "path": "plugin/src/main/groovy/com/github/gfx/ribbonizer/GrayScaleBuilder.java",
    "content": "package com.github.gfx.ribbonizer;\n\nimport com.android.build.gradle.api.ApplicationVariant;\nimport com.github.gfx.ribbonizer.filter.GrayScaleFilter;\n\nimport java.awt.image.BufferedImage;\nimport java.io.File;\nimport java.util.function.Consumer;\n\npublic class GrayScaleBuilder implements FilterBuilder {\n\n    @Override\n    public Consumer<BufferedImage> apply(ApplicationVariant applicationVariant, File iconFile) {\n        return new GrayScaleFilter();\n    }\n}\n"
  },
  {
    "path": "plugin/src/main/groovy/com/github/gfx/ribbonizer/GreenRibbonBuilder.java",
    "content": "package com.github.gfx.ribbonizer;\n\nimport com.android.build.gradle.api.ApplicationVariant;\nimport com.github.gfx.ribbonizer.filter.ColorRibbonFilter;\n\nimport java.awt.*;\nimport java.awt.image.BufferedImage;\nimport java.io.File;\nimport java.util.function.Consumer;\n\npublic class GreenRibbonBuilder implements FilterBuilder {\n\n    public static Color COLOR = new Color(0, 0x72, 0, 0x99);\n\n    @Override\n    public Consumer<BufferedImage> apply(ApplicationVariant variant, File iconFile) {\n        return new ColorRibbonFilter(variant.getBuildType().getName(), COLOR);\n    }\n}\n"
  },
  {
    "path": "plugin/src/main/groovy/com/github/gfx/ribbonizer/YellowRibbonBuilder.java",
    "content": "package com.github.gfx.ribbonizer;\n\nimport com.android.build.gradle.api.ApplicationVariant;\nimport com.github.gfx.ribbonizer.filter.ColorRibbonFilter;\n\nimport java.awt.*;\nimport java.awt.image.BufferedImage;\nimport java.io.File;\nimport java.util.function.Consumer;\n\npublic class YellowRibbonBuilder implements FilterBuilder {\n\n    public static final Color COLOR = new Color(0xff, 0x76, 0, 0x99);\n\n    @Override\n    public Consumer<BufferedImage> apply(ApplicationVariant variant, File iconFile) {\n        return new ColorRibbonFilter(variant.getBuildType().getName(), COLOR);\n    }\n}\n"
  },
  {
    "path": "plugin/src/main/groovy/com/github/gfx/ribbonizer/filter/ColorRibbonFilter.java",
    "content": "package com.github.gfx.ribbonizer.filter;\n\nimport java.awt.*;\nimport java.awt.font.FontRenderContext;\nimport java.awt.geom.AffineTransform;\nimport java.awt.geom.Rectangle2D;\nimport java.awt.image.BufferedImage;\nimport java.util.function.Consumer;\n\npublic class ColorRibbonFilter implements Consumer<BufferedImage> {\n\n    static final boolean debug = Boolean.parseBoolean(System.getenv(\"RIBBONIZER_DEBUG\"));\n\n    final Color ribbonColor;\n\n    final Color labelColor;\n\n    String label;\n\n    String fontName = \"Default\";\n\n    int fontStyle = Font.PLAIN;\n\n    boolean largeRibbon = false;\n\n    public ColorRibbonFilter(String label, Color ribbonColor, Color labelColor) {\n        this.label = label;\n        this.ribbonColor = ribbonColor;\n        this.labelColor = labelColor;\n    }\n\n    public ColorRibbonFilter(String label, Color ribbonColor) {\n        this(label, ribbonColor, Color.WHITE);\n    }\n\n    static int calculateMaxLabelWidth(int y) {\n        return (int) Math.sqrt(Math.pow(y, 2) * 2);\n    }\n\n    static void drawString(Graphics2D g, String str, int x, int y) {\n        g.drawString(str, x, y);\n\n        if (debug) {\n            FontMetrics fm = g.getFontMetrics();\n            Rectangle2D bounds = g.getFont().getStringBounds(str,\n                    new FontRenderContext(g.getTransform(), true, true));\n\n            g.drawRect(x, y - fm.getAscent(), (int) bounds.getWidth(), fm.getAscent());\n        }\n    }\n\n    @Override\n    public void accept(BufferedImage image) {\n        int width = image.getWidth();\n        int height = image.getHeight();\n\n        Graphics2D g = (Graphics2D) image.getGraphics();\n\n        g.setTransform(AffineTransform.getRotateInstance(Math.toRadians(-45)));\n\n        int y = height / (largeRibbon ? 2 : 4);\n\n        // calculate the rectangle where the label is rendered\n        FontRenderContext frc = new FontRenderContext(g.getTransform(), true, true);\n        int maxLabelWidth = calculateMaxLabelWidth(y);\n        g.setFont(getFont(maxLabelWidth, frc));\n        Rectangle2D labelBounds = g.getFont().getStringBounds(label == null ? \"\" : label, frc);\n\n        // draw the ribbon\n        g.setColor(ribbonColor);\n        g.fillRect(-width, y, width * 2, (int) (labelBounds.getHeight()));\n\n        if (label != null) {\n            // draw the label\n            g.setRenderingHint(RenderingHints.KEY_ANTIALIASING,\n                    RenderingHints.VALUE_ANTIALIAS_ON);\n            g.setColor(labelColor);\n\n            FontMetrics fm = g.getFontMetrics();\n\n            drawString(g, label,\n                    (int) -labelBounds.getWidth() / 2,\n                    y + fm.getAscent());\n        }\n        g.dispose();\n    }\n\n    Font getFont(int maxLabelWidth, FontRenderContext frc) {\n        int max = 32;\n        if (label == null) {\n            return new Font(fontName, fontStyle, max / 2);\n        }\n        int min = 0;\n        int x = max;\n\n        for (int i = 0; i < 10; i++) {\n            int m = ((max + min) / 2);\n            if (m == x) {\n                break;\n            }\n\n            Font font = new Font(fontName, fontStyle, m);\n            Rectangle2D labelBounds = font.getStringBounds(label, frc);\n            int px = (int) labelBounds.getWidth();\n\n            if (px > maxLabelWidth) {\n                max = m;\n            } else {\n                min = m;\n            }\n            x = m;\n        }\n        return new Font(fontName, fontStyle, x);\n    }\n}"
  },
  {
    "path": "plugin/src/main/groovy/com/github/gfx/ribbonizer/filter/GrayScaleFilter.java",
    "content": "package com.github.gfx.ribbonizer.filter;\n\nimport java.awt.image.BufferedImage;\nimport java.util.function.Consumer;\n\npublic class GrayScaleFilter implements Consumer<BufferedImage> {\n\n    static int toGray(int color) {\n        int a = (color & 0xFF000000);\n        int r = (color & 0x00FF0000) >> 16;\n        int g = (color & 0x0000FF00) >> 8;\n        int b = (color & 0x000000FF);\n\n        int c = (int) ((2.0 * r + 4.0 * g + b) / 7.0);\n        return a | (c << 16) | (c << 8) | c;\n    }\n\n    @Override\n    public void accept(BufferedImage image) {\n        int width = image.getWidth();\n        int height = image.getHeight();\n\n        for (int y = 0; y < height; y++) {\n            for (int x = 0; x < width; x++) {\n                int color = image.getRGB(x, y);\n                image.setRGB(x, y, toGray(color));\n            }\n        }\n    }\n\n}\n"
  },
  {
    "path": "plugin/src/main/groovy/com/github/gfx/ribbonizer/plugin/Resources.java",
    "content": "package com.github.gfx.ribbonizer.plugin;\n\nimport org.xml.sax.SAXException;\n\nimport java.io.File;\nimport java.io.IOException;\nimport java.util.ArrayList;\nimport java.util.List;\n\nimport javax.xml.parsers.ParserConfigurationException;\n\nimport groovy.util.XmlSlurper;\nimport groovy.util.slurpersupport.GPathResult;\n\nimport static java.util.Collections.unmodifiableList;\n\npublic class Resources {\n\n    public static String resourceFilePattern(String name) {\n        if (name.startsWith(\"@\")) {\n            String[] pair = name.substring(1).split(\"/\", 2);\n            String baseResType = pair[0];\n            String fileName = pair[1];\n            if (fileName == null) {\n                throw new IllegalArgumentException(\n                        \"Icon names does include resource types (e.g. drawable/ic_launcher): \"\n                                + name);\n            }\n            return baseResType + \"*/\" + fileName + \".*\";\n        } else {\n            return name;\n        }\n    }\n\n    public static List<String> getLauncherIcons(File manifestFile)\n            throws SAXException, ParserConfigurationException, IOException {\n        GPathResult manifestXml = new XmlSlurper().parse(manifestFile);\n        GPathResult applicationNode = (GPathResult) manifestXml.getProperty(\"application\");\n\n        String icon = String.valueOf(applicationNode.getProperty(\"@android:icon\"));\n        String roundIcon = String.valueOf(applicationNode.getProperty(\"@android:roundIcon\"));\n\n        List<String> icons = new ArrayList<>(2);\n        if (!icon.isEmpty()) {\n            icons.add(icon);\n        }\n        if (!roundIcon.isEmpty()) {\n            icons.add(roundIcon);\n        }\n\n        return unmodifiableList(icons);\n\n    }\n}\n"
  },
  {
    "path": "plugin/src/main/groovy/com/github/gfx/ribbonizer/plugin/Ribbonizer.java",
    "content": "package com.github.gfx.ribbonizer.plugin;\n\nimport java.awt.image.BufferedImage;\nimport java.io.File;\nimport java.io.IOException;\nimport java.util.function.Consumer;\nimport java.util.stream.Stream;\n\nimport javax.imageio.ImageIO;\n\npublic class Ribbonizer {\n\n    final File inputFile;\n\n    final File outputFile;\n\n    final BufferedImage image;\n\n    public Ribbonizer(File inputFile, File outputFile) throws IOException {\n        this.inputFile = inputFile;\n        this.outputFile = outputFile;\n\n        image = ImageIO.read(inputFile);\n    }\n\n    public void save() throws IOException {\n        outputFile.getParentFile().mkdirs();\n        ImageIO.write(image, \"png\", outputFile);\n    }\n\n    public void process(Stream<Consumer<BufferedImage>> filters) {\n        filters.forEach(new Consumer<Consumer<BufferedImage>>() {\n            @Override\n            public void accept(Consumer<BufferedImage> filter) {\n                if (filter != null) {\n                    filter.accept(image);\n                }\n            }\n        });\n    }\n}"
  },
  {
    "path": "plugin/src/main/groovy/com/github/gfx/ribbonizer/plugin/RibbonizerExtension.java",
    "content": "package com.github.gfx.ribbonizer.plugin;\n\nimport com.android.build.gradle.api.ApplicationVariant;\nimport com.github.gfx.ribbonizer.CustomColorRibbonBuilder;\nimport com.github.gfx.ribbonizer.FilterBuilder;\nimport com.github.gfx.ribbonizer.GrayScaleBuilder;\nimport com.github.gfx.ribbonizer.GrayRibbonBuilder;\nimport com.github.gfx.ribbonizer.GreenRibbonBuilder;\nimport com.github.gfx.ribbonizer.YellowRibbonBuilder;\n\nimport java.awt.image.BufferedImage;\nimport java.io.File;\nimport java.util.ArrayList;\nimport java.util.Arrays;\nimport java.util.Collection;\nimport java.util.HashSet;\nimport java.util.List;\nimport java.util.Set;\nimport java.util.function.Consumer;\n\nclass RibbonizerExtension {\n\n    public static String NAME = \"ribbonizer\";\n\n    Set<String> forcedVariantsNames = new HashSet<>();\n\n    Set<String> iconNames = new HashSet<>();\n\n    List<FilterBuilder> filterBuilders = new ArrayList<>();\n\n    public RibbonizerExtension() {\n    }\n\n    public Set<String> getIconNames() {\n        return iconNames;\n    }\n\n    /**\n     * @param resNames Names of icons. For example \"@drawable/ic_launcher\", \"@mipmap/icon\"\n     */\n    public void setIconNames(Collection<String> resNames) {\n        iconNames = new HashSet<>(resNames);\n    }\n\n    /**\n     * @param resNames Names of icons. For example \"@drawable/ic_launcher\", \"@mipmap/icon\"\n     */\n    public void iconNames(Collection<String> resNames) {\n        setIconNames(resNames);\n    }\n\n    /**\n     * @param resNames Names of icons. For example \"@drawable/ic_launcher\", \"@mipmap/icon\"\n     */\n    public void iconNames(String... resNames) {\n        setIconNames(Arrays.asList(resNames));\n    }\n\n    /**\n     * @param resName A name of icons. For example \"@drawable/ic_launcher\", \"@mipmap/icon\"\n     */\n    public void iconName(String resName) {\n        iconNames.add(resName);\n    }\n\n    public Set<String> getForcedVariantsNames() {\n        return forcedVariantsNames;\n    }\n\n    public void forcedVariantsNames(String... variantsNames) {\n        forcedVariantsNames = new HashSet<>(Arrays.asList(variantsNames));\n    }\n\n    public List<FilterBuilder> getFilterBuilders() {\n        return filterBuilders;\n    }\n\n    public void setFilterBuilders(Collection<FilterBuilder> filterBuilders) {\n        this.filterBuilders = new ArrayList<>(filterBuilders);\n    }\n\n    public void builder(FilterBuilder filterBuilder)\n            throws IllegalAccessException, InstantiationException {\n        this.filterBuilders.clear();\n        this.filterBuilders.add(filterBuilder);\n    }\n\n    // utilities\n\n    public Consumer<BufferedImage> grayScaleFilter(ApplicationVariant variant, File iconFile) {\n        return new GrayScaleBuilder().apply(variant, iconFile);\n    }\n\n    public Consumer<BufferedImage> grayRibbonFilter(ApplicationVariant variant, File iconFile) {\n        return new GrayRibbonBuilder().apply(variant, iconFile);\n    }\n\n    public Consumer<BufferedImage> yellowRibbonFilter(ApplicationVariant variant, File iconFile) {\n        return new YellowRibbonBuilder().apply(variant, iconFile);\n    }\n\n    public Consumer<BufferedImage> greenRibbonFilter(ApplicationVariant variant, File iconFile) {\n        return new GreenRibbonBuilder().apply(variant, iconFile);\n    }\n\n    public Consumer<BufferedImage> customColorRibbonFilter(ApplicationVariant variant, File iconFile, String nm) {\n        return new CustomColorRibbonBuilder(nm).apply(variant, iconFile);\n    }\n}\n"
  },
  {
    "path": "plugin/src/main/groovy/com/github/gfx/ribbonizer/plugin/RibbonizerPlugin.groovy",
    "content": "package com.github.gfx.ribbonizer.plugin\n\nimport com.android.build.gradle.AppExtension\nimport com.android.build.gradle.api.ApplicationVariant\nimport com.github.gfx.ribbonizer.FilterBuilder\nimport com.github.gfx.ribbonizer.GreenRibbonBuilder\nimport groovy.transform.CompileStatic\nimport org.gradle.api.Plugin\nimport org.gradle.api.Project\nimport org.gradle.api.Task\n\n// see http://www.gradle.org/docs/current/userguide/custom_plugins.html\n\n@CompileStatic\npublic class RibbonizerPlugin implements Plugin<Project> {\n\n    static {\n        System.setProperty(\"java.awt.headless\", \"true\")\n\n        // workaround for an Android Studio issue\n        try {\n            Class.forName(System.getProperty(\"java.awt.graphicsenv\"))\n        } catch (ClassNotFoundException e) {\n            System.err.println(\"[WARN] java.awt.graphicsenv: \" + e)\n            System.setProperty(\"java.awt.graphicsenv\", \"sun.awt.CGraphicsEnvironment\")\n        }\n        try {\n            Class.forName(System.getProperty(\"awt.toolkit\"))\n        } catch (ClassNotFoundException e) {\n            System.err.println(\"[WARN] awt.toolkit: \" + e)\n            System.setProperty(\"awt.toolkit\", \"sun.lwawt.macosx.LWCToolkit\")\n        }\n    }\n\n    @Override\n    void apply(Project project) {\n        project.extensions.add(RibbonizerExtension.NAME, RibbonizerExtension)\n\n        project.afterEvaluate {\n            def android = project.extensions.findByType(AppExtension)\n            if (!android) {\n                throw new Exception(\n                        \"Not an Android application; you forget `apply plugin: 'com.android.application`?\")\n            }\n            def extension = project.extensions.findByType(RibbonizerExtension)\n\n            def tasks = new ArrayList<Task>();\n\n            android.applicationVariants.all { ApplicationVariant variant ->\n                if ((!variant.buildType.debuggable) && (!extension.forcedVariantsNames.contains(variant.name))) {\n                    project.logger.info(\"[ribbonizer] skip ${variant.name} because it is not debuggable and not forced.\")\n                    return;\n                }\n\n                List<FilterBuilder> filterBuilders = extension.filterBuilders\n                if (filterBuilders.size() == 0) {\n                    filterBuilders = [new GreenRibbonBuilder() as FilterBuilder]\n                }\n\n                def generatedResDir = getGeneratedResDir(project, variant)\n                android.sourceSets.findByName(variant.name).res.srcDir(generatedResDir)\n\n                def name = \"${RibbonizerTask.NAME}${capitalize(variant.name)}\"\n                def task = project.task(name, type: RibbonizerTask) as RibbonizerTask\n                task.variant = variant\n                task.outputDir = generatedResDir\n                task.iconNames = new HashSet<String>(extension.iconNames)\n                task.filterBuilders = filterBuilders\n                tasks.add(task)\n\n                def generateResources = project.\n                        getTasksByName(\"generate${capitalize(variant.name)}Resources\", false)\n                generateResources.forEach { Task t ->\n                    t.dependsOn(task)\n                }\n            }\n\n            project.task(RibbonizerTask.NAME, dependsOn: tasks);\n        }\n    }\n\n    static String capitalize(String str) {\n        return str.substring(0, 1).toUpperCase() + str.substring(1);\n    }\n\n    static File getGeneratedResDir(Project project, ApplicationVariant variant) {\n        return new File(project.buildDir,\n                \"generated/ribbonizer/res/${variant.name}\")\n    }\n}\n"
  },
  {
    "path": "plugin/src/main/groovy/com/github/gfx/ribbonizer/plugin/RibbonizerTask.groovy",
    "content": "package com.github.gfx.ribbonizer.plugin\n\nimport com.android.build.gradle.AppExtension\nimport com.android.build.gradle.api.ApplicationVariant\nimport com.android.builder.model.SourceProvider\nimport com.github.gfx.ribbonizer.FilterBuilder\nimport org.gradle.api.DefaultTask\nimport org.gradle.api.tasks.TaskAction\n\nimport java.awt.image.BufferedImage\nimport java.util.function.Consumer\nimport java.util.function.Function\nimport java.util.stream.Stream\n\nclass RibbonizerTask extends DefaultTask {\n\n    static final String NAME = \"ribbonize\"\n\n    ApplicationVariant variant\n\n    //@OutputDirectory\n    File outputDir\n\n    // `iconNames` includes: \"@drawable/icon\", \"@mipmap/ic_launcher\", etc.\n    Set<String> iconNames\n\n    List<FilterBuilder> filterBuilders = []\n\n    @TaskAction\n    public void run() {\n        if (filterBuilders.size() == 0) {\n            return;\n        }\n\n        def t0 = System.currentTimeMillis()\n\n        def names = new HashSet<String>(iconNames)\n        names.addAll(launcherIconNames)\n\n        variant.sourceSets.stream()\n                .flatMap(new Function<SourceProvider, Stream>() {\n\n            @Override\n            Stream apply(SourceProvider sourceProvider) {\n                return sourceProvider.resDirectories.stream()\n            }\n        }).forEach { File resDir ->\n            if (resDir == outputDir) {\n                return\n            }\n\n            names.forEach { String name ->\n                project.fileTree(\n                        dir: resDir,\n                        include: Resources.resourceFilePattern(name),\n                        exclude: \"**/*.xml\",\n                ).forEach { File inputFile ->\n                    info \"process $inputFile\"\n\n                    def basename = inputFile.name\n                    def resType = inputFile.parentFile.name\n                    def outputFile = new File(outputDir, \"${resType}/${basename}\")\n                    outputFile.parentFile.mkdirs()\n\n                    def ribbonizer = new Ribbonizer(inputFile, outputFile)\n                    ribbonizer.process(filterBuilders.stream()\n                            .map(new Function<FilterBuilder, Consumer<BufferedImage>>() {\n\n                        @Override\n                        Consumer<BufferedImage> apply(FilterBuilder filterBuilder) {\n                            return filterBuilder.apply(variant, inputFile)\n                        }\n                    }))\n                    ribbonizer.save()\n                }\n            }\n        }\n\n        info(\"task finished in ${System.currentTimeMillis() - t0}ms\")\n    }\n\n    void info(String message) {\n        //System.out.println(\"[$name] $message\")\n        project.logger.info(\"[$name] $message\")\n    }\n\n    Set<String> getLauncherIconNames() {\n        def names = new HashSet<String>()\n        androidManifestFiles.forEach { File manifestFile ->\n            names.addAll(Resources.getLauncherIcons(manifestFile))\n        }\n        return names\n    }\n\n    Stream<File> getAndroidManifestFiles() {\n        AppExtension android = project.extensions.findByType(AppExtension)\n\n        return [\"main\", variant.name, variant.buildType.name, variant.flavorName].stream()\n                .filter({ name -> !name.empty })\n                .distinct()\n                .map({ name -> project.file(android.sourceSets[name].manifest.srcFile) })\n                .filter({ manifestFile -> manifestFile.exists() })\n    }\n}"
  },
  {
    "path": "plugin/src/main/resources/META-INF/gradle-plugins/com.github.gfx.ribbonizer.properties",
    "content": "implementation-class=com.github.gfx.ribbonizer.plugin.RibbonizerPlugin"
  },
  {
    "path": "plugin/src/test/groovy/com/github/gfx/ribbonizer/test/ResourcesTest.groovy",
    "content": "package com.github.gfx.ribbonizer.test\n\nimport com.github.gfx.ribbonizer.plugin.Resources\nimport spock.lang.Specification\n\npublic class ResourcesTest extends Specification {\n    def \"resourceFilePattern\"() {\n        expect:\n        Resources.resourceFilePattern(resName) == pattern\n\n        where:\n        resName                 | pattern\n        \"@drawable/ic_launcher\" | \"drawable*/ic_launcher.*\"\n        \"@mipmap/icon\"          | \"mipmap*/icon.*\"\n    }\n\n    def \"getLauncherIcon without android:roundIcon\"() {\n        setup:\n        def file = File.createTempFile(\"AndroidManifest\", \".xml\")\n        file.deleteOnExit()\n        file.write('''\n<?xml version=\"1.0\" encoding=\"utf-8\"?>\n<manifest xmlns:android=\"http://schemas.android.com/apk/res/android\"\n    package=\"com.github.gfx.ribbonizer.example\"\n    android:versionCode=\"1\"\n    android:versionName=\"1.0\" >\n    <uses-sdk\n        android:minSdkVersion=\"15\"\n        android:targetSdkVersion=\"23\" />\n    <application\n        android:allowBackup=\"true\"\n        android:icon=\"@drawable/ic_launcher\"\n        android:label=\"@string/app_name\"\n        android:theme=\"@style/AppTheme\" >\n    </application>\n</manifest>\n'''.trim())\n\n        expect:\n        Resources.getLauncherIcons(file).containsAll([\"@drawable/ic_launcher\"])\n    }\n\n    def \"getLauncherIcon without android:icon\"() {\n        setup:\n        def file = File.createTempFile(\"AndroidManifest\", \".xml\")\n        file.deleteOnExit()\n        file.write('''\n<?xml version=\"1.0\" encoding=\"utf-8\"?>\n<manifest xmlns:android=\"http://schemas.android.com/apk/res/android\"\n    package=\"com.github.gfx.ribbonizer.example\"\n    android:versionCode=\"1\"\n    android:versionName=\"1.0\" >\n    <uses-sdk\n        android:minSdkVersion=\"15\"\n        android:targetSdkVersion=\"23\" />\n    <application\n        android:allowBackup=\"true\"\n        android:roundIcon=\"@drawable/ic_launcher_round\"\n        android:label=\"@string/app_name\"\n        android:theme=\"@style/AppTheme\" >\n    </application>\n</manifest>\n'''.trim())\n\n        expect:\n        Resources.getLauncherIcons(file).containsAll([\"@drawable/ic_launcher_round\"])\n    }\n\n    def \"getLauncherIcon with both android:icon and android:roundIcon\"() {\n        setup:\n        def file = File.createTempFile(\"AndroidManifest\", \".xml\")\n        file.deleteOnExit()\n        file.write('''\n<?xml version=\"1.0\" encoding=\"utf-8\"?>\n<manifest xmlns:android=\"http://schemas.android.com/apk/res/android\"\n    package=\"com.github.gfx.ribbonizer.example\"\n    android:versionCode=\"1\"\n    android:versionName=\"1.0\" >\n    <uses-sdk\n        android:minSdkVersion=\"15\"\n        android:targetSdkVersion=\"23\" />\n    <application\n        android:allowBackup=\"true\"\n        android:icon=\"@drawable/ic_launcher\"\n        android:roundIcon=\"@drawable/ic_launcher_round\"\n        android:label=\"@string/app_name\"\n        android:theme=\"@style/AppTheme\" >\n    </application>\n</manifest>\n'''.trim())\n\n        expect:\n        Resources.getLauncherIcons(file).containsAll([\"@drawable/ic_launcher\", \"@drawable/ic_launcher_round\"])\n    }\n}\n"
  },
  {
    "path": "settings.gradle",
    "content": "include ':plugin'\ninclude ':example-simple'\ninclude ':example-custom'\n"
  },
  {
    "path": "versioning.gradle",
    "content": "/*\n * Copyright (c) 2015 FUJI Goro (gfx).\n *\n * Licensed under the Apache License, Version 2.0 (the \"License\");\n * you may not use this file except in compliance with the License.\n * You may obtain a copy of the License at\n *\n * http://www.apache.org/licenses/LICENSE-2.0\n *\n * Unless required by applicable law or agreed to in writing, software\n * distributed under the License is distributed on an \"AS IS\" BASIS,\n * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.\n * See the License for the specific language governing permissions and\n * limitations under the License.\n */\n\nimport java.util.regex.Pattern\n\nstatic int parseVersionName(String versionName) {\n    final versionParts = versionName.split(/\\D/)\n    final major = versionParts[0].toInteger()\n    final minor = versionParts[1].toInteger()\n    final patchLevel = versionParts[2].toInteger()\n\n    return major * (1000 * 1000) + minor * 1000 + patchLevel;\n}\n\nvoid updateReadme(String oldVersion, String newVersion) {\n    def template = rootProject.file('README.md').text\n    def result = template.replaceAll(Pattern.quote(oldVersion), newVersion)\n    rootProject.file(\"README.md\").withWriter { it << result }\n}\n\nrootProject.ext {\n    versionFile = rootProject.file(\"VERSION\")\n    versionName = versionFile.readLines()[0].trim()\n    versionCode = parseVersionName(versionName)\n}\n\n\ntask bumpMajor {\n    doLast {\n        def oldVersion = rootProject.ext.versionName\n        def versionParts = oldVersion.split(/\\./)\n        versionParts[0] = (versionParts[0] as int) + 1\n        def newVersion = \"${versionParts[0]}.0.0\"\n        versionFile.write(newVersion + \"\\n\")\n        updateReadme(oldVersion, newVersion)\n        rootProject.ext.versionName = newVersion\n        tasks.version.execute()\n    }\n}\n\ntask bumpMinor {\n    doLast {\n        def oldVersion = rootProject.ext.versionName\n        def versionParts = oldVersion.split(/\\./)\n        versionParts[1] = (versionParts[1] as int) + 1\n        def newVersion = \"${versionParts[0]}.${versionParts[1]}.0\"\n        versionFile.write(newVersion + \"\\n\")\n        updateReadme(oldVersion, newVersion)\n        rootProject.ext.versionName = newVersion\n        tasks.version.execute()\n    }\n}\n\ntask bumpPatch {\n    doLast {\n        def oldVersion = rootProject.ext.versionName\n        def versionParts = oldVersion.split(/\\./)\n        versionParts[2] = (versionParts[2].split(/\\D/)[0] as int) + 1\n        def newVersion = \"${versionParts[0]}.${versionParts[1]}.${versionParts[2]}\"\n        versionFile.write(newVersion + \"\\n\")\n        updateReadme(oldVersion, newVersion)\n        rootProject.ext.versionName = newVersion\n        tasks.version.execute()\n    }\n}\n\ntask version {\n    doLast {\n        println \"v\" + rootProject.ext.versionName\n    }\n}\n\nString shell(String command) {\n    def proc = [\"sh\", \"-c\", \"cd ${rootProject.rootDir} ; ${command}\"].execute()\n    if (proc.waitFor() != 0) {\n        throw new RuntimeException(\"Failed to run: ${command}\\n${proc.err.text}\")\n    } else {\n        def err = proc.err.text\n        if (err) {\n            System.err.println(err)\n        }\n    }\n    return proc.in.text;\n}\n\ntask releng {\n    doLast {\n        def currentBranch = shell(\"git symbolic-ref --short HEAD\")\n        def tag = \"v\" + rootProject.ext.versionName\n        println \"Release engineering for ${tag} (branch=${currentBranch})\"\n\n        def changes = shell(\"git status -s\")\n        if (changes.trim()) {\n            throw new RuntimeException(\"There are changes not commited yet.\\n${changes}\")\n        }\n\n        println \"> git tag ${tag}\"\n        shell \"git tag ${tag}\"\n\n        println \"> git push origin ${tag}\"\n        shell \"git push origin ${tag}\"\n\n        println \"> git push origin ${tag}\"\n        shell \"git push origin ${currentBranch}\"\n    }\n}\n"
  }
]