Moved to Android Studio.

Added theme to Google accounts selection dialog.
This commit is contained in:
Erik C. Thauvin 2015-04-14 11:39:19 -07:00
parent 3d9d98e11f
commit 1cfaa3b523
66 changed files with 1503 additions and 733 deletions

15
.gitignore vendored
View file

@ -1,10 +1,7 @@
/.classpath
/.idea/libraries
.gradle
/local.properties
/.idea/workspace.xml
/.pmd
/.project
/.svn/
/bin
/gen
/proguard
/project.properties
/.idea/libraries
.DS_Store
/build
/captures

1
.idea/.name generated Normal file
View file

@ -0,0 +1 @@
Emaily

201
.idea/codeStyleSettings.xml generated Normal file
View file

@ -0,0 +1,201 @@
<?xml version="1.0" encoding="UTF-8"?>
<project version="4">
<component name="ProjectCodeStyleSettingsManager">
<option name="PER_PROJECT_SETTINGS">
<value>
<option name="CLASS_COUNT_TO_USE_IMPORT_ON_DEMAND" value="99" />
<option name="NAMES_COUNT_TO_USE_IMPORT_ON_DEMAND" value="99" />
<option name="PACKAGES_TO_USE_IMPORT_ON_DEMAND">
<value />
</option>
<option name="IMPORT_LAYOUT_TABLE">
<value>
<package name="android" withSubpackages="true" static="false" />
<emptyLine />
<package name="com" withSubpackages="true" static="false" />
<emptyLine />
<package name="junit" withSubpackages="true" static="false" />
<emptyLine />
<package name="net" withSubpackages="true" static="false" />
<emptyLine />
<package name="org" withSubpackages="true" static="false" />
<emptyLine />
<package name="java" withSubpackages="true" static="false" />
<emptyLine />
<package name="javax" withSubpackages="true" static="false" />
<emptyLine />
<package name="" withSubpackages="true" static="false" />
<emptyLine />
<package name="" withSubpackages="true" static="true" />
<emptyLine />
</value>
</option>
<option name="RIGHT_MARGIN" value="100" />
<AndroidXmlCodeStyleSettings>
<option name="USE_CUSTOM_SETTINGS" value="true" />
</AndroidXmlCodeStyleSettings>
<XML>
<option name="XML_LEGACY_SETTINGS_IMPORTED" value="true" />
</XML>
<codeStyleSettings language="XML">
<option name="FORCE_REARRANGE_MODE" value="1" />
<indentOptions>
<option name="CONTINUATION_INDENT_SIZE" value="4" />
</indentOptions>
<arrangement>
<rules>
<section>
<rule>
<match>
<AND>
<NAME>xmlns:android</NAME>
<XML_NAMESPACE>Namespace:</XML_NAMESPACE>
</AND>
</match>
</rule>
</section>
<section>
<rule>
<match>
<AND>
<NAME>xmlns:.*</NAME>
<XML_NAMESPACE>Namespace:</XML_NAMESPACE>
</AND>
</match>
<order>BY_NAME</order>
</rule>
</section>
<section>
<rule>
<match>
<AND>
<NAME>.*:id</NAME>
<XML_NAMESPACE>http://schemas.android.com/apk/res/android</XML_NAMESPACE>
</AND>
</match>
</rule>
</section>
<section>
<rule>
<match>
<AND>
<NAME>.*:name</NAME>
<XML_NAMESPACE>http://schemas.android.com/apk/res/android</XML_NAMESPACE>
</AND>
</match>
</rule>
</section>
<section>
<rule>
<match>
<AND>
<NAME>name</NAME>
<XML_NAMESPACE>^$</XML_NAMESPACE>
</AND>
</match>
</rule>
</section>
<section>
<rule>
<match>
<AND>
<NAME>style</NAME>
<XML_NAMESPACE>^$</XML_NAMESPACE>
</AND>
</match>
</rule>
</section>
<section>
<rule>
<match>
<AND>
<NAME>.*</NAME>
<XML_NAMESPACE>^$</XML_NAMESPACE>
</AND>
</match>
<order>BY_NAME</order>
</rule>
</section>
<section>
<rule>
<match>
<AND>
<NAME>.*:layout_width</NAME>
<XML_NAMESPACE>http://schemas.android.com/apk/res/android</XML_NAMESPACE>
</AND>
</match>
</rule>
</section>
<section>
<rule>
<match>
<AND>
<NAME>.*:layout_height</NAME>
<XML_NAMESPACE>http://schemas.android.com/apk/res/android</XML_NAMESPACE>
</AND>
</match>
</rule>
</section>
<section>
<rule>
<match>
<AND>
<NAME>.*:layout_.*</NAME>
<XML_NAMESPACE>http://schemas.android.com/apk/res/android</XML_NAMESPACE>
</AND>
</match>
<order>BY_NAME</order>
</rule>
</section>
<section>
<rule>
<match>
<AND>
<NAME>.*:width</NAME>
<XML_NAMESPACE>http://schemas.android.com/apk/res/android</XML_NAMESPACE>
</AND>
</match>
<order>BY_NAME</order>
</rule>
</section>
<section>
<rule>
<match>
<AND>
<NAME>.*:height</NAME>
<XML_NAMESPACE>http://schemas.android.com/apk/res/android</XML_NAMESPACE>
</AND>
</match>
<order>BY_NAME</order>
</rule>
</section>
<section>
<rule>
<match>
<AND>
<NAME>.*</NAME>
<XML_NAMESPACE>http://schemas.android.com/apk/res/android</XML_NAMESPACE>
</AND>
</match>
<order>BY_NAME</order>
</rule>
</section>
<section>
<rule>
<match>
<AND>
<NAME>.*</NAME>
<XML_NAMESPACE>.*</XML_NAMESPACE>
</AND>
</match>
<order>BY_NAME</order>
</rule>
</section>
</rules>
</arrangement>
</codeStyleSettings>
</value>
</option>
<option name="PREFERRED_PROJECT_CODE_STYLE" value="Default (1)" />
</component>
</project>

22
.idea/compiler.xml generated Normal file
View file

@ -0,0 +1,22 @@
<?xml version="1.0" encoding="UTF-8"?>
<project version="4">
<component name="CompilerConfiguration">
<option name="DEFAULT_COMPILER" value="Javac" />
<resourceExtensions />
<wildcardResourcePatterns>
<entry name="!?*.java" />
<entry name="!?*.form" />
<entry name="!?*.class" />
<entry name="!?*.groovy" />
<entry name="!?*.scala" />
<entry name="!?*.flex" />
<entry name="!?*.kt" />
<entry name="!?*.clj" />
</wildcardResourcePatterns>
<annotationProcessing>
<profile default="true" name="Default" enabled="false">
<processorPath useClasspath="true" />
</profile>
</annotationProcessing>
</component>
</project>

3
.idea/copyright/profiles_settings.xml generated Normal file
View file

@ -0,0 +1,3 @@
<component name="CopyrightManager">
<settings default="" />
</component>

18
.idea/gradle.xml generated Normal file
View file

@ -0,0 +1,18 @@
<?xml version="1.0" encoding="UTF-8"?>
<project version="4">
<component name="GradleSettings">
<option name="linkedExternalProjectsSettings">
<GradleProjectSettings>
<option name="distributionType" value="DEFAULT_WRAPPED" />
<option name="externalProjectPath" value="$PROJECT_DIR$" />
<option name="gradleJvm" value="1.8" />
<option name="modules">
<set>
<option value="$PROJECT_DIR$" />
<option value="$PROJECT_DIR$/app" />
</set>
</option>
</GradleProjectSettings>
</option>
</component>
</project>

View file

@ -0,0 +1,37 @@
<component name="InspectionProjectProfileManager">
<profile version="1.0">
<option name="myName" value="Project Default" />
<option name="myLocal" value="true" />
<inspection_tool class="JavaDoc" enabled="true" level="WARNING" enabled_by_default="true">
<option name="TOP_LEVEL_CLASS_OPTIONS">
<value>
<option name="ACCESS_JAVADOC_REQUIRED_FOR" value="none" />
<option name="REQUIRED_TAGS" value="" />
</value>
</option>
<option name="INNER_CLASS_OPTIONS">
<value>
<option name="ACCESS_JAVADOC_REQUIRED_FOR" value="none" />
<option name="REQUIRED_TAGS" value="" />
</value>
</option>
<option name="METHOD_OPTIONS">
<value>
<option name="ACCESS_JAVADOC_REQUIRED_FOR" value="none" />
<option name="REQUIRED_TAGS" value="@return@param@throws or @exception" />
</value>
</option>
<option name="FIELD_OPTIONS">
<value>
<option name="ACCESS_JAVADOC_REQUIRED_FOR" value="none" />
<option name="REQUIRED_TAGS" value="" />
</value>
</option>
<option name="IGNORE_DEPRECATED" value="false" />
<option name="IGNORE_JAVADOC_PERIOD" value="true" />
<option name="IGNORE_DUPLICATED_THROWS" value="false" />
<option name="IGNORE_POINT_TO_ITSELF" value="false" />
<option name="myAdditionalJavadocTags" value="created" />
</inspection_tool>
</profile>
</component>

View file

@ -0,0 +1,7 @@
<component name="InspectionProjectProfileManager">
<settings>
<option name="PROJECT_PROFILE" value="Project Default" />
<option name="USE_PROJECT_PROFILE" value="true" />
<version value="1.0" />
</settings>
</component>

22
.idea/misc.xml generated Normal file
View file

@ -0,0 +1,22 @@
<?xml version="1.0" encoding="UTF-8"?>
<project version="4">
<component name="EntryPointsManager">
<entry_points version="2.0" />
</component>
<component name="ProjectLevelVcsManager" settingsEditedManually="false">
<OptionsSetting value="true" id="Add" />
<OptionsSetting value="true" id="Remove" />
<OptionsSetting value="true" id="Checkout" />
<OptionsSetting value="true" id="Update" />
<OptionsSetting value="true" id="Status" />
<OptionsSetting value="true" id="Edit" />
<ConfirmationsSetting value="0" id="Add" />
<ConfirmationsSetting value="0" id="Remove" />
</component>
<component name="ProjectRootManager" version="2" languageLevel="JDK_1_7" assert-keyword="true" jdk-15="true" project-jdk-name="1.8" project-jdk-type="JavaSDK">
<output url="file://$PROJECT_DIR$/build/classes" />
</component>
<component name="ProjectType">
<option name="id" value="Android" />
</component>
</project>

9
.idea/modules.xml generated Normal file
View file

@ -0,0 +1,9 @@
<?xml version="1.0" encoding="UTF-8"?>
<project version="4">
<component name="ProjectModuleManager">
<modules>
<module fileurl="file://$PROJECT_DIR$/Emaily.iml" filepath="$PROJECT_DIR$/Emaily.iml" />
<module fileurl="file://$PROJECT_DIR$/app/app.iml" filepath="$PROJECT_DIR$/app/app.iml" />
</modules>
</component>
</project>

6
.idea/vcs.xml generated Normal file
View file

@ -0,0 +1,6 @@
<?xml version="1.0" encoding="UTF-8"?>
<project version="4">
<component name="VcsDirectoryMappings">
<mapping directory="$PROJECT_DIR$" vcs="Git" />
</component>
</project>

View file

@ -1,4 +0,0 @@
eclipse.preferences.version=1
org.eclipse.jdt.core.compiler.codegen.targetPlatform=1.6
org.eclipse.jdt.core.compiler.compliance=1.6
org.eclipse.jdt.core.compiler.source=1.6

19
Emaily.iml Normal file
View file

@ -0,0 +1,19 @@
<?xml version="1.0" encoding="UTF-8"?>
<module external.linked.project.id="Emaily" external.linked.project.path="$MODULE_DIR$" external.root.project.path="$MODULE_DIR$" external.system.id="GRADLE" external.system.module.group="" external.system.module.version="unspecified" type="JAVA_MODULE" version="4">
<component name="FacetManager">
<facet type="java-gradle" name="Java-Gradle">
<configuration>
<option name="BUILD_FOLDER_PATH" value="$MODULE_DIR$/build" />
<option name="BUILDABLE" value="false" />
</configuration>
</facet>
</component>
<component name="NewModuleRootManager" inherit-compiler-output="true">
<exclude-output />
<content url="file://$MODULE_DIR$">
<excludeFolder url="file://$MODULE_DIR$/.gradle" />
</content>
<orderEntry type="inheritedJdk" />
<orderEntry type="sourceFolder" forTests="false" />
</component>
</module>

1
app/.gitignore vendored Normal file
View file

@ -0,0 +1 @@
/build

103
app/app.iml Normal file
View file

@ -0,0 +1,103 @@
<?xml version="1.0" encoding="UTF-8"?>
<module external.linked.project.id=":app" external.linked.project.path="$MODULE_DIR$" external.root.project.path="$MODULE_DIR$/.." external.system.id="GRADLE" external.system.module.group="Emaily" external.system.module.version="unspecified" type="JAVA_MODULE" version="4">
<component name="FacetManager">
<facet type="android-gradle" name="Android-Gradle">
<configuration>
<option name="GRADLE_PROJECT_PATH" value=":app" />
</configuration>
</facet>
<facet type="android" name="Android">
<configuration>
<option name="SELECTED_BUILD_VARIANT" value="debug" />
<option name="SELECTED_TEST_ARTIFACT" value="_android_test_" />
<option name="ASSEMBLE_TASK_NAME" value="assembleDebug" />
<option name="COMPILE_JAVA_TASK_NAME" value="compileDebugSources" />
<option name="SOURCE_GEN_TASK_NAME" value="generateDebugSources" />
<option name="ASSEMBLE_TEST_TASK_NAME" value="assembleDebugAndroidTest" />
<option name="COMPILE_JAVA_TEST_TASK_NAME" value="compileDebugAndroidTestSources" />
<option name="TEST_SOURCE_GEN_TASK_NAME" value="generateDebugAndroidTestSources" />
<option name="ALLOW_USER_CONFIGURATION" value="false" />
<option name="MANIFEST_FILE_RELATIVE_PATH" value="/src/main/AndroidManifest.xml" />
<option name="RES_FOLDER_RELATIVE_PATH" value="/src/main/res" />
<option name="RES_FOLDERS_RELATIVE_PATH" value="file://$MODULE_DIR$/src/main/res" />
<option name="ASSETS_FOLDER_RELATIVE_PATH" value="/src/main/assets" />
</configuration>
</facet>
</component>
<component name="NewModuleRootManager" inherit-compiler-output="false">
<output url="file://$MODULE_DIR$/build/intermediates/classes/debug" />
<output-test url="file://$MODULE_DIR$/build/intermediates/classes/androidTest/debug" />
<exclude-output />
<content url="file://$MODULE_DIR$">
<sourceFolder url="file://$MODULE_DIR$/build/generated/source/r/debug" isTestSource="false" generated="true" />
<sourceFolder url="file://$MODULE_DIR$/build/generated/source/aidl/debug" isTestSource="false" generated="true" />
<sourceFolder url="file://$MODULE_DIR$/build/generated/source/buildConfig/debug" isTestSource="false" generated="true" />
<sourceFolder url="file://$MODULE_DIR$/build/generated/source/rs/debug" isTestSource="false" generated="true" />
<sourceFolder url="file://$MODULE_DIR$/build/generated/res/rs/debug" type="java-resource" />
<sourceFolder url="file://$MODULE_DIR$/build/generated/res/generated/debug" type="java-resource" />
<sourceFolder url="file://$MODULE_DIR$/build/generated/source/r/androidTest/debug" isTestSource="true" generated="true" />
<sourceFolder url="file://$MODULE_DIR$/build/generated/source/aidl/androidTest/debug" isTestSource="true" generated="true" />
<sourceFolder url="file://$MODULE_DIR$/build/generated/source/buildConfig/androidTest/debug" isTestSource="true" generated="true" />
<sourceFolder url="file://$MODULE_DIR$/build/generated/source/rs/androidTest/debug" isTestSource="true" generated="true" />
<sourceFolder url="file://$MODULE_DIR$/build/generated/res/rs/androidTest/debug" type="java-test-resource" />
<sourceFolder url="file://$MODULE_DIR$/build/generated/res/generated/androidTest/debug" type="java-test-resource" />
<sourceFolder url="file://$MODULE_DIR$/src/debug/res" type="java-resource" />
<sourceFolder url="file://$MODULE_DIR$/src/debug/resources" type="java-resource" />
<sourceFolder url="file://$MODULE_DIR$/src/debug/assets" type="java-resource" />
<sourceFolder url="file://$MODULE_DIR$/src/debug/aidl" isTestSource="false" />
<sourceFolder url="file://$MODULE_DIR$/src/debug/java" isTestSource="false" />
<sourceFolder url="file://$MODULE_DIR$/src/debug/jni" isTestSource="false" />
<sourceFolder url="file://$MODULE_DIR$/src/debug/rs" isTestSource="false" />
<sourceFolder url="file://$MODULE_DIR$/src/main/res" type="java-resource" />
<sourceFolder url="file://$MODULE_DIR$/src/main/resources" type="java-resource" />
<sourceFolder url="file://$MODULE_DIR$/src/main/assets" type="java-resource" />
<sourceFolder url="file://$MODULE_DIR$/src/main/aidl" isTestSource="false" />
<sourceFolder url="file://$MODULE_DIR$/src/main/java" isTestSource="false" />
<sourceFolder url="file://$MODULE_DIR$/src/main/jni" isTestSource="false" />
<sourceFolder url="file://$MODULE_DIR$/src/main/rs" isTestSource="false" />
<sourceFolder url="file://$MODULE_DIR$/src/androidTest/res" type="java-test-resource" />
<sourceFolder url="file://$MODULE_DIR$/src/androidTest/resources" type="java-test-resource" />
<sourceFolder url="file://$MODULE_DIR$/src/androidTest/assets" type="java-test-resource" />
<sourceFolder url="file://$MODULE_DIR$/src/androidTest/aidl" isTestSource="true" />
<sourceFolder url="file://$MODULE_DIR$/src/androidTest/java" isTestSource="true" />
<sourceFolder url="file://$MODULE_DIR$/src/androidTest/jni" isTestSource="true" />
<sourceFolder url="file://$MODULE_DIR$/src/androidTest/rs" isTestSource="true" />
<excludeFolder url="file://$MODULE_DIR$/build/intermediates/assets" />
<excludeFolder url="file://$MODULE_DIR$/build/intermediates/bundles" />
<excludeFolder url="file://$MODULE_DIR$/build/intermediates/classes" />
<excludeFolder url="file://$MODULE_DIR$/build/intermediates/coverage-instrumented-classes" />
<excludeFolder url="file://$MODULE_DIR$/build/intermediates/dependency-cache" />
<excludeFolder url="file://$MODULE_DIR$/build/intermediates/dex" />
<excludeFolder url="file://$MODULE_DIR$/build/intermediates/dex-cache" />
<excludeFolder url="file://$MODULE_DIR$/build/intermediates/incremental" />
<excludeFolder url="file://$MODULE_DIR$/build/intermediates/jacoco" />
<excludeFolder url="file://$MODULE_DIR$/build/intermediates/javaResources" />
<excludeFolder url="file://$MODULE_DIR$/build/intermediates/libs" />
<excludeFolder url="file://$MODULE_DIR$/build/intermediates/lint" />
<excludeFolder url="file://$MODULE_DIR$/build/intermediates/manifests" />
<excludeFolder url="file://$MODULE_DIR$/build/intermediates/ndk" />
<excludeFolder url="file://$MODULE_DIR$/build/intermediates/pre-dexed" />
<excludeFolder url="file://$MODULE_DIR$/build/intermediates/proguard" />
<excludeFolder url="file://$MODULE_DIR$/build/intermediates/res" />
<excludeFolder url="file://$MODULE_DIR$/build/intermediates/rs" />
<excludeFolder url="file://$MODULE_DIR$/build/intermediates/symbols" />
<excludeFolder url="file://$MODULE_DIR$/build/outputs" />
<excludeFolder url="file://$MODULE_DIR$/build/tmp" />
</content>
<orderEntry type="jdk" jdkName="Android API 22 Platform (1)" jdkType="Android SDK" />
<orderEntry type="sourceFolder" forTests="false" />
<orderEntry type="library" exported="" name="gson-2.1" level="project" />
<orderEntry type="library" exported="" name="protobuf-java-2.2.0" level="project" />
<orderEntry type="library" exported="" name="google-http-client-1.7.0-beta" level="project" />
<orderEntry type="library" exported="" name="google-api-client-1.7.0-beta" level="project" />
<orderEntry type="library" exported="" name="bitlyj-2.0.0" level="project" />
<orderEntry type="library" exported="" name="google-http-client-android2-1.7.0-beta" level="project" />
<orderEntry type="library" exported="" name="google-oauth-client-1.7.0-beta" level="project" />
<orderEntry type="library" exported="" name="jsr305-1.3.9" level="project" />
<orderEntry type="library" exported="" name="google-api-client-android2-1.7.0-beta" level="project" />
<orderEntry type="library" exported="" name="jackson-core-asl-1.9.4" level="project" />
<orderEntry type="library" exported="" name="google-api-urlshortener-v1-rev2-java-1.4.0-beta" level="project" />
<orderEntry type="library" exported="" name="guava-11.0.1" level="project" />
<orderEntry type="library" exported="" name="google-http-client-android3-1.7.0-beta" level="project" />
</component>
</module>

25
app/build.gradle Normal file
View file

@ -0,0 +1,25 @@
apply plugin: 'com.android.application'
android {
compileSdkVersion 22
buildToolsVersion "21.1.2"
defaultConfig {
applicationId "net.thauvin.erik.android.emaily"
minSdkVersion 11
targetSdkVersion 22
versionCode 2
versionName "1.1b9"
}
buildTypes {
release {
minifyEnabled true
proguardFiles getDefaultProguardFile('proguard-android.txt'), 'proguard-rules.pro'
}
}
}
dependencies {
compile fileTree(dir: 'libs', include: ['*.jar'])
//compile 'com.android.support:appcompat-v7:22.0.0'
}

View file

@ -1,3 +1,20 @@
# Add project specific ProGuard rules here.
# By default, the flags in this file are appended to flags specified
# in C:\Users\erik\AppData\Local\Android\android-sdk/tools/proguard/proguard-android.txt
# You can edit the include path and order by changing the proguardFiles
# directive in build.gradle.
#
# For more details, see
# http://developer.android.com/guide/developing/tools/proguard.html
# Add any project specific keep options here:
# If your project uses WebView with JS, uncomment the following
# and specify the fully qualified class name to the JavaScript interface
# class:
#-keepclassmembers class fqcn.of.javascript.interface.for.webview {
# public *;
#}
-optimizationpasses 5
-dontusemixedcaseclassnames
-dontskipnonpubliclibraryclasses

View file

@ -1,12 +1,6 @@
<?xml version="1.0" encoding="utf-8"?>
<manifest xmlns:android="http://schemas.android.com/apk/res/android"
package="net.thauvin.erik.android.emaily"
android:versionCode="2"
android:versionName="1.1b8" >
<uses-sdk
android:minSdkVersion="10"
android:targetSdkVersion="21" />
package="net.thauvin.erik.android.emaily">
<uses-permission android:name="android.permission.INTERNET" />
<uses-permission android:name="android.permission.GET_ACCOUNTS" />
@ -14,13 +8,13 @@
<application
android:allowBackup="true"
android:icon="@drawable/icon"
android:icon="@mipmap/icon"
android:label="@string/app_name"
android:theme="@style/Emaily" >
android:theme="@style/Emaily">
<activity
android:name="net.thauvin.erik.android.emaily.Emaily"
android:label="@string/app_name"
android:theme="@android:style/Theme.Translucent.NoTitleBar.Fullscreen" >
android:theme="@android:style/Theme.Translucent.NoTitleBar.Fullscreen">
<intent-filter>
<action android:name="android.intent.action.SEND" />
@ -31,7 +25,7 @@
</activity>
<activity
android:name=".EmailyPrefs"
android:label="@string/app_name" >
android:label="@string/app_name">
<intent-filter>
<action android:name="android.intent.action.MAIN" />

View file

@ -87,7 +87,7 @@ public class BitlyCredsDialog extends DialogPreference
{
final Intent intent = new Intent(Intent.ACTION_VIEW, Uri.parse(mContext.getString(R.string.prefs_bitly_creds_url)));
mContext.startActivity(intent);
};
}
});
}

View file

@ -0,0 +1,699 @@
/*
* @(#)Emaily.java
*
* Copyright (c) 2011-2012 Erik C. Thauvin (http://erik.thauvin.net/)
* All rights reserved.
*
* Redistribution and use in source and binary forms, with or without
* modification, are permitted provided that the following conditions are
* met:
*
* Redistributions of source code must retain the above copyright notice,
* this list of conditions and the following disclaimer.
*
* Redistributions in binary form must reproduce the above copyright notice,
* this list of conditions and the following disclaimer in the documentation
* and/or other materials provided with the distribution.
*
* Neither the name of the authors nor the names of its contributors may be
* used to endorse or promote products derived from this software without
* specific prior written permission.
*
* THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS
* IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO,
* THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR
* PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT OWNER OR
* CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL,
* EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO,
* PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR
* PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF
* LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING
* NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS
* SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
*
* $Id$
*
*/
package net.thauvin.erik.android.emaily;
import static com.rosaloves.bitlyj.Bitly.as;
import static com.rosaloves.bitlyj.Bitly.shorten;
import java.io.IOException;
import java.net.MalformedURLException;
import java.net.URL;
import java.net.UnknownHostException;
import java.util.Date;
import android.accounts.Account;
import android.accounts.AccountManager;
import android.accounts.AccountManagerCallback;
import android.accounts.AccountManagerFuture;
import android.accounts.OperationCanceledException;
import android.app.Activity;
import android.app.AlertDialog;
import android.app.ProgressDialog;
import android.content.Context;
import android.content.DialogInterface;
import android.content.Intent;
import android.content.SharedPreferences;
import android.content.SharedPreferences.Editor;
import android.content.pm.PackageManager.NameNotFoundException;
import android.os.AsyncTask;
import android.os.Bundle;
import android.preference.PreferenceManager;
import android.text.ClipboardManager;
import android.text.Html;
import android.util.Log;
import android.widget.Toast;
import com.google.api.client.googleapis.extensions.android2.auth.GoogleAccountManager;
import com.google.api.client.googleapis.json.GoogleJsonError;
import com.google.api.client.googleapis.json.GoogleJsonResponseException;
import com.google.api.client.http.HttpTransport;
import com.google.api.client.http.javanet.NetHttpTransport;
import com.google.api.client.http.json.JsonHttpRequest;
import com.google.api.client.http.json.JsonHttpRequestInitializer;
import com.google.api.client.json.JsonFactory;
import com.google.api.client.json.jackson.JacksonFactory;
import com.google.api.services.urlshortener.Urlshortener;
import com.google.api.services.urlshortener.UrlshortenerRequest;
import com.google.api.services.urlshortener.model.Url;
/**
* The <code>Emaily</code> class implements a URL shortener intent.
*
* @author <a href="mailto:erik@thauvin.net">Erik C. Thauvin</a>
* @version $Revision$
* @created Oct 11, 2011
* @since 1.0
*/
@SuppressWarnings("deprecation")
public class Emaily extends Activity
{
private static final String ACCOUNT_TYPE = "com.google";
private static final String OAUTH_URL = "oauth2:https://www.googleapis.com/auth/urlshortener";
private String appName;
private SharedPreferences sharedPrefs;
@Override
public void onCreate(Bundle savedInstanceState)
{
super.onCreate(savedInstanceState);
final Intent intent = getIntent();
appName = getString(R.string.app_name);
sharedPrefs = PreferenceManager.getDefaultSharedPreferences(this);
if (Intent.ACTION_SEND.equals(intent.getAction()))
{
final boolean isGoogl = getBoolPref(R.string.prefs_key_googl_enabled, true);
if (isGoogl)
{
final String account = getPref(R.string.prefs_key_googl_account);
if (isValid(account))
{
startEmailyTask(intent, new Account(account, ACCOUNT_TYPE), false);
}
else
{
final AlertDialog.Builder builder = new AlertDialog.Builder(this, AlertDialog.THEME_DEVICE_DEFAULT_LIGHT);
builder.setTitle(R.string.dialog_accounts_title);
final Account[] accounts = AccountManager.get(this).getAccountsByType(ACCOUNT_TYPE);
final int size = accounts.length;
if (size > 0)
{
if (size == 1)
{
startEmailyTask(intent, accounts[0], false);
}
else
{
final CharSequence[] names = new CharSequence[size];
for (int i = 0; i < size; i++)
{
names[i] = accounts[i].name;
}
builder.setSingleChoiceItems(names, 0, new DialogInterface.OnClickListener()
{
@Override
public void onClick(DialogInterface dialog, int which)
{
dialog.dismiss();
final Editor editor = sharedPrefs.edit();
editor.putString(getString(R.string.prefs_key_googl_account), names[which].toString());
editor.putLong(getString(R.string.prefs_key_googl_token_expiry), 0L);
editor.commit();
startEmailyTask(intent, accounts[which], false);
}
});
builder.create().show();
}
}
else
{
startEmailyTask(intent, isGoogl, false);
}
}
}
else
{
startEmailyTask(intent, isGoogl, false);
}
}
else
{
Emaily.this.finish();
}
}
/**
* Starts the task.
*
* @param intent The original intent.
* @param isGoogl The goo.gl flag.
* @param isRetry The retry flag.
*/
private void startEmailyTask(final Intent intent, final boolean isGoogl, final boolean isRetry)
{
final EmailyTask task;
if (isGoogl)
{
task = new EmailyTask(getPref(R.string.prefs_key_googl_account), getPref(R.string.prefs_key_googl_token), isGoogl,
getBoolPref(R.string.prefs_key_html_chkbox), isRetry);
}
else
{
task = new EmailyTask(getPref(R.string.prefs_key_bitly_username), getPref(R.string.prefs_key_bitly_apikey), isGoogl,
getBoolPref(R.string.prefs_key_html_chkbox), isRetry);
}
task.execute(intent);
}
/**
* Starts the task.
*
* @param intent The original intent.
* @param account The account.
* @param isRetry The retry flag.
*/
private void startEmailyTask(final Intent intent, final Account account, final boolean isRetry)
{
final GoogleAccountManager googleAccountManager = new GoogleAccountManager(Emaily.this);
final long expiry = sharedPrefs.getLong(getString(R.string.prefs_key_googl_token_expiry), 0L);
final long now = System.currentTimeMillis();
final long maxLife = (60L * 55L) * 1000L; // 55 minutes
Log.d(appName, "Token Expires: " + new Date(expiry));
if (expiry >= (now + maxLife) || expiry <= now)
{
final String token = getPref(R.string.prefs_key_googl_token);
if (isValid(token))
{
googleAccountManager.manager.invalidateAuthToken(ACCOUNT_TYPE, token);
Log.d(appName, "Token Invalidated: " + token);
}
}
googleAccountManager.manager.getAuthToken(account, OAUTH_URL, null, Emaily.this, new AccountManagerCallback<Bundle>()
{
@Override
public void run(AccountManagerFuture<Bundle> future)
{
try
{
final String token = future.getResult().getString(AccountManager.KEY_AUTHTOKEN);
final Editor editor = sharedPrefs.edit();
final long now = System.currentTimeMillis();
final long expires;
if (expiry < now)
{
expires = now + maxLife;
}
else
{
expires = expiry;
}
editor.putLong(getString(R.string.prefs_key_googl_token_expiry), expires);
editor.putString(getString(R.string.prefs_key_googl_token), token);
editor.commit();
Log.d(appName, account.toString());
Log.d(appName, "Token: " + token);
Log.d(appName, "Expires: " + new Date(expires));
startEmailyTask(intent, true, isRetry);
}
catch (OperationCanceledException e)
{
Log.e(appName, "Auth token request has been canceled.", e);
}
catch (Exception e)
{
Log.e(appName, "Exception while requesting the auth token.", e);
}
}
}, null);
}
/**
* Retries the task.
*
* @param intent The original intent.
*/
public void retry(final Intent intent)
{
sharedPrefs.edit().putLong(getString(R.string.prefs_key_googl_token_expiry), 0L).commit();
startEmailyTask(intent, new Account(getPref(R.string.prefs_key_googl_account), ACCOUNT_TYPE), true);
}
/**
* Validates a string.
*
* @param s The string to validate.
* @return returns <code>true</code> if the string is not empty or null, <code>false</code> otherwise.
*/
public static boolean isValid(String s)
{
return (s != null) && (!s.trim().isEmpty());
}
/**
* Returns the value of the specified shared reference based on the specified string id. The default value is an empty string.
*
* @param id The string id.
* @return The preference value.
*/
public String getPref(int id)
{
return getPref(id, "");
}
/**
* Returns the value of the specified shared reference based on the specified string id.
*
* @param id The string id.
* @param defaultValue The default value, used if the preference is empty.
* @return The preference value.
*/
public String getPref(int id, String defaultValue)
{
return sharedPrefs.getString(getString(id), defaultValue);
}
/**
* Returns the value of the specified shared reference based on the specified string id. The default value is <code>false</code>.
*
* @param id The string id.
* @return The preference value.
*/
public boolean getBoolPref(int id)
{
return getBoolPref(id, false);
}
/**
* Returns the value of the specified shared reference based on the specified string id.
*
* @param id The string id.
* @param defaultValue The default value, used if the preference is empty.
* @return The preference value.
*/
public boolean getBoolPref(int id, boolean defaultValue)
{
return sharedPrefs.getBoolean(getString(id), defaultValue);
}
/**
* The <code>EmailyTask</code> class.
*/
private class EmailyTask extends AsyncTask<Intent, Void, EmailyResult>
{
private final ProgressDialog dialog = new ProgressDialog(Emaily.this);
private final String username;
private final String keytoken;
private final boolean isGoogl;
private final boolean isHtml;
private final boolean isRetry;
public EmailyTask(String username, String keytoken, boolean isGoogl, boolean isHtml, boolean isRetry)
{
this.username = username;
this.keytoken = keytoken;
this.isGoogl = isGoogl;
this.isHtml = isHtml;
this.isRetry = isRetry;
}
@Override
protected EmailyResult doInBackground(Intent... intent)
{
final EmailyResult result = new EmailyResult(intent[0]);
final Intent emailIntent = new Intent(Intent.ACTION_SEND);
if (isHtml)
{
emailIntent.setType("text/html");
}
else
{
emailIntent.setType("text/plain");
}
final Bundle extras = intent[0].getExtras();
final String pageUrl = extras.getString(Intent.EXTRA_TEXT);
final String pageTitle = extras.getString(Intent.EXTRA_SUBJECT);
final StringBuilder textBefore = new StringBuilder();
if (isValid(pageTitle))
{
emailIntent.putExtra(Intent.EXTRA_SUBJECT, pageTitle);
}
final boolean hasCredentials = isValid(username) && isValid(keytoken);
final StringBuilder shortUrl = new StringBuilder();
if (isValid(pageUrl))
{
final HttpTransport transport = new NetHttpTransport();
final JsonFactory jsonFactory = new JacksonFactory();
String version = "";
try
{
version = '/' + getPackageManager().getPackageInfo(getPackageName(), 0).versionName;
}
catch (NameNotFoundException ignore)
{
// Do nothing;
}
final Url toInsert = new Url();
final String[] splits = pageUrl.split("\\s");
for (String item : splits)
{
try
{
new URL(item.trim());
if (isGoogl || !hasCredentials)
{
Log.d(appName, "goo.gl -> " + item);
final Urlshortener shortener = Urlshortener.builder(transport, jsonFactory).setApplicationName(appName + version)
.setJsonHttpRequestInitializer(new JsonHttpRequestInitializer()
{
@Override
public void initialize(JsonHttpRequest request) throws IOException
{
UrlshortenerRequest shortnerRequest = (UrlshortenerRequest) request;
shortnerRequest.setKey(getString(R.string.secret_apikey));
if (isValid(keytoken))
{
shortnerRequest.setOauthToken(keytoken);
}
shortnerRequest.put("client_id", getString(R.string.secret_client_id));
shortnerRequest.put("client_secret", getString(R.string.secret_client_secret));
}
}).build();
toInsert.setLongUrl(item.trim());
try
{
final Url shortened = shortener.url().insert(toInsert).execute();
shortUrl.append(shortened.getId());
}
catch (GoogleJsonResponseException e)
{
result.setCode(R.string.alert_error);
final GoogleJsonError err = e.getDetails();
result.setMessage(err.message);
if (err.code == 401)
{
if (!isRetry)
{
result.setRetry(true);
}
}
Log.e(appName, "Exception while shortening '" + item + "' via goo.gl.", e);
}
catch (UnknownHostException e)
{
result.setCode(R.string.alert_nohost);
result.setMessage(e.getMessage());
Log.e(appName, "UnknownHostException while shortening '" + item + "' via goo.gl.", e);
}
catch (IOException e)
{
result.setCode(R.string.alert_error);
result.setMessage(e.getMessage());
Log.e(appName, "IOException while shortening '" + item + "' via goo.gl.", e);
}
}
else
{
Log.d(appName, "bit.ly -> " + item);
try
{
shortUrl.append(as(username, keytoken).call(shorten(item.trim())).getShortUrl());
}
catch (Exception e)
{
final Throwable cause = e.getCause();
if (cause != null && cause instanceof UnknownHostException)
{
result.setCode(R.string.alert_nohost);
result.setMessage(cause.getMessage());
}
else
{
result.setCode(R.string.alert_error);
result.setMessage(e.getMessage());
}
Log.e(appName, "Exception while shortening '" + item + "' via bit.ly.", e);
}
break;
}
break;
}
catch (MalformedURLException mue)
{
Log.d(appName, "Attempted to process an invalid URL: " + item, mue);
if (textBefore.length() > 0)
{
textBefore.append(" ");
}
textBefore.append(item);
}
}
}
else
{
result.setCode(R.string.alert_nocreds);
}
if (!result.isRetry())
{
if (shortUrl.length() > 0)
{
if (isHtml)
{
emailIntent.putExtra(Intent.EXTRA_TEXT, Html.fromHtml("<a href=\"" + shortUrl + "\">" + shortUrl + "</a>"));
}
else
{
emailIntent.putExtra(Intent.EXTRA_TEXT, shortUrl.toString());
}
if (!isValid(pageTitle) && textBefore.length() > 0)
{
emailIntent.putExtra(Intent.EXTRA_SUBJECT, textBefore.toString());
}
}
else
{
final CharSequence chars = extras.getCharSequence(Intent.EXTRA_TEXT);
if (chars.length() > 0)
{
emailIntent.putExtra(Intent.EXTRA_TEXT, chars);
}
else if (isValid(pageUrl))
{
emailIntent.putExtra(Intent.EXTRA_TEXT, pageUrl);
}
}
try
{
startActivity(emailIntent);
}
catch (android.content.ActivityNotFoundException ignore)
{
if (!result.hasError() && shortUrl.length() > 0)
{
final ClipboardManager clip = (ClipboardManager) getSystemService(Context.CLIPBOARD_SERVICE);
clip.setText(shortUrl);
result.setCode(R.string.alert_notfound_clip);
}
else
{
result.setCode(R.string.alert_notfound);
}
}
}
return result;
}
@Override
protected void onPostExecute(EmailyResult result)
{
if (this.dialog.isShowing())
{
this.dialog.dismiss();
}
if (result.isRetry())
{
Emaily.this.retry(result.getIntent());
}
else
{
if (result.hasError())
{
Toast.makeText(
getApplicationContext(),
getString(result.getCode(), result.getMessage(), isGoogl ? getString(R.string.prefs_googl_title)
: getString(R.string.prefs_bitly_title)), Toast.LENGTH_LONG).show();
}
Emaily.this.finish();
}
}
@Override
protected void onPreExecute()
{
if (isRetry)
{
this.dialog.setMessage(getString(R.string.progress_msg_retry));
}
else
{
this.dialog.setMessage(getString(R.string.progress_msg));
}
this.dialog.show();
}
}
/**
* The <code>EmailyResult</code> class.
*/
private class EmailyResult
{
private int code = 0;
private String message;
private boolean retry = false;
private final Intent intent;
public EmailyResult(Intent intent)
{
this.intent = intent;
}
public int getCode()
{
return code;
}
public String getMessage()
{
if (isValid(message))
{
return message;
}
else
{
return "";
}
}
public Intent getIntent()
{
return intent;
}
public boolean hasError()
{
return code != 0;
}
public boolean isRetry()
{
return retry;
}
public void setCode(int code)
{
this.code = code;
}
public void setMessage(String message)
{
this.message = message;
}
public void setRetry(boolean retry)
{
this.retry = retry;
}
}
}

View file

@ -88,7 +88,7 @@ public class EmailyPrefs extends PreferenceActivity implements OnSharedPreferenc
mBitlyCreds.setEnabled(false);
}
final Preference version = (Preference) findPreference(getString(R.string.prefs_key_version));
final Preference version = findPreference(getString(R.string.prefs_key_version));
final PreferenceScreen feedback = (PreferenceScreen) findPreference(getString(R.string.prefs_key_feedback));
try
{

View file

Before

Width:  |  Height:  |  Size: 3.9 KiB

After

Width:  |  Height:  |  Size: 3.9 KiB

Before After
Before After

View file

Before

Width:  |  Height:  |  Size: 1.6 KiB

After

Width:  |  Height:  |  Size: 1.6 KiB

Before After
Before After

View file

Before

Width:  |  Height:  |  Size: 2.2 KiB

After

Width:  |  Height:  |  Size: 2.2 KiB

Before After
Before After

View file

Before

Width:  |  Height:  |  Size: 5.6 KiB

After

Width:  |  Height:  |  Size: 5.6 KiB

Before After
Before After

View file

@ -1 +1 @@
/secret.xml
/secret.xml

View file

@ -19,7 +19,7 @@
<string name="prefs_bitly_creds_url">http://bitly.com/a/your_api_key/</string>
<string name="prefs_bitly_creds_username">Username</string>
<string name="prefs_bitly_title">bit.ly</string>
<string name="prefs_copyright">© 2012-14 Erik C. Thauvin</string>
<string name="prefs_copyright">© 201214 Erik C. Thauvin</string>
<string name="prefs_feedback_subject">%1$s %2$s %3$s (%4$s %5$s, %6$s)</string>
<string name="prefs_feedback_summary">Send email, please check Help first…</string>
<string name="prefs_feedback_title">Feedback</string>

19
build.gradle Normal file
View file

@ -0,0 +1,19 @@
// Top-level build file where you can add configuration options common to all sub-projects/modules.
buildscript {
repositories {
jcenter()
}
dependencies {
classpath 'com.android.tools.build:gradle:1.1.0'
// NOTE: Do not place your application dependencies here; they belong
// in the individual module build.gradle files
}
}
allprojects {
repositories {
jcenter()
}
}

18
gradle.properties Normal file
View file

@ -0,0 +1,18 @@
# Project-wide Gradle settings.
# IDE (e.g. Android Studio) users:
# Gradle settings configured through the IDE *will override*
# any settings specified in this file.
# For more details on how to configure your build environment visit
# http://www.gradle.org/docs/current/userguide/build_environment.html
# Specifies the JVM arguments used for the daemon process.
# The setting is particularly useful for tweaking memory settings.
# Default value: -Xmx10248m -XX:MaxPermSize=256m
# org.gradle.jvmargs=-Xmx2048m -XX:MaxPermSize=512m -XX:+HeapDumpOnOutOfMemoryError -Dfile.encoding=UTF-8
# When configured, Gradle will run in incubating parallel mode.
# This option should only be used with decoupled projects. More details, visit
# http://www.gradle.org/docs/current/userguide/multi_project_builds.html#sec:decoupled_projects
# org.gradle.parallel=true

BIN
gradle/wrapper/gradle-wrapper.jar vendored Normal file

Binary file not shown.

View file

@ -0,0 +1,6 @@
#Wed Apr 10 15:27:10 PDT 2013
distributionBase=GRADLE_USER_HOME
distributionPath=wrapper/dists
zipStoreBase=GRADLE_USER_HOME
zipStorePath=wrapper/dists
distributionUrl=https\://services.gradle.org/distributions/gradle-2.2.1-all.zip

164
gradlew vendored Normal file
View file

@ -0,0 +1,164 @@
#!/usr/bin/env bash
##############################################################################
##
## Gradle start up script for UN*X
##
##############################################################################
# Add default JVM options here. You can also use JAVA_OPTS and GRADLE_OPTS to pass JVM options to this script.
DEFAULT_JVM_OPTS=""
APP_NAME="Gradle"
APP_BASE_NAME=`basename "$0"`
# Use the maximum available, or set MAX_FD != -1 to use that value.
MAX_FD="maximum"
warn ( ) {
echo "$*"
}
die ( ) {
echo
echo "$*"
echo
exit 1
}
# OS specific support (must be 'true' or 'false').
cygwin=false
msys=false
darwin=false
case "`uname`" in
CYGWIN* )
cygwin=true
;;
Darwin* )
darwin=true
;;
MINGW* )
msys=true
;;
esac
# For Cygwin, ensure paths are in UNIX format before anything is touched.
if $cygwin ; then
[ -n "$JAVA_HOME" ] && JAVA_HOME=`cygpath --unix "$JAVA_HOME"`
fi
# Attempt to set APP_HOME
# Resolve links: $0 may be a link
PRG="$0"
# Need this for relative symlinks.
while [ -h "$PRG" ] ; do
ls=`ls -ld "$PRG"`
link=`expr "$ls" : '.*-> \(.*\)$'`
if expr "$link" : '/.*' > /dev/null; then
PRG="$link"
else
PRG=`dirname "$PRG"`"/$link"
fi
done
SAVED="`pwd`"
cd "`dirname \"$PRG\"`/" >&-
APP_HOME="`pwd -P`"
cd "$SAVED" >&-
CLASSPATH=$APP_HOME/gradle/wrapper/gradle-wrapper.jar
# Determine the Java command to use to start the JVM.
if [ -n "$JAVA_HOME" ] ; then
if [ -x "$JAVA_HOME/jre/sh/java" ] ; then
# IBM's JDK on AIX uses strange locations for the executables
JAVACMD="$JAVA_HOME/jre/sh/java"
else
JAVACMD="$JAVA_HOME/bin/java"
fi
if [ ! -x "$JAVACMD" ] ; then
die "ERROR: JAVA_HOME is set to an invalid directory: $JAVA_HOME
Please set the JAVA_HOME variable in your environment to match the
location of your Java installation."
fi
else
JAVACMD="java"
which java >/dev/null 2>&1 || die "ERROR: JAVA_HOME is not set and no 'java' command could be found in your PATH.
Please set the JAVA_HOME variable in your environment to match the
location of your Java installation."
fi
# Increase the maximum file descriptors if we can.
if [ "$cygwin" = "false" -a "$darwin" = "false" ] ; then
MAX_FD_LIMIT=`ulimit -H -n`
if [ $? -eq 0 ] ; then
if [ "$MAX_FD" = "maximum" -o "$MAX_FD" = "max" ] ; then
MAX_FD="$MAX_FD_LIMIT"
fi
ulimit -n $MAX_FD
if [ $? -ne 0 ] ; then
warn "Could not set maximum file descriptor limit: $MAX_FD"
fi
else
warn "Could not query maximum file descriptor limit: $MAX_FD_LIMIT"
fi
fi
# For Darwin, add options to specify how the application appears in the dock
if $darwin; then
GRADLE_OPTS="$GRADLE_OPTS \"-Xdock:name=$APP_NAME\" \"-Xdock:icon=$APP_HOME/media/gradle.icns\""
fi
# For Cygwin, switch paths to Windows format before running java
if $cygwin ; then
APP_HOME=`cygpath --path --mixed "$APP_HOME"`
CLASSPATH=`cygpath --path --mixed "$CLASSPATH"`
# We build the pattern for arguments to be converted via cygpath
ROOTDIRSRAW=`find -L / -maxdepth 1 -mindepth 1 -type d 2>/dev/null`
SEP=""
for dir in $ROOTDIRSRAW ; do
ROOTDIRS="$ROOTDIRS$SEP$dir"
SEP="|"
done
OURCYGPATTERN="(^($ROOTDIRS))"
# Add a user-defined pattern to the cygpath arguments
if [ "$GRADLE_CYGPATTERN" != "" ] ; then
OURCYGPATTERN="$OURCYGPATTERN|($GRADLE_CYGPATTERN)"
fi
# Now convert the arguments - kludge to limit ourselves to /bin/sh
i=0
for arg in "$@" ; do
CHECK=`echo "$arg"|egrep -c "$OURCYGPATTERN" -`
CHECK2=`echo "$arg"|egrep -c "^-"` ### Determine if an option
if [ $CHECK -ne 0 ] && [ $CHECK2 -eq 0 ] ; then ### Added a condition
eval `echo args$i`=`cygpath --path --ignore --mixed "$arg"`
else
eval `echo args$i`="\"$arg\""
fi
i=$((i+1))
done
case $i in
(0) set -- ;;
(1) set -- "$args0" ;;
(2) set -- "$args0" "$args1" ;;
(3) set -- "$args0" "$args1" "$args2" ;;
(4) set -- "$args0" "$args1" "$args2" "$args3" ;;
(5) set -- "$args0" "$args1" "$args2" "$args3" "$args4" ;;
(6) set -- "$args0" "$args1" "$args2" "$args3" "$args4" "$args5" ;;
(7) set -- "$args0" "$args1" "$args2" "$args3" "$args4" "$args5" "$args6" ;;
(8) set -- "$args0" "$args1" "$args2" "$args3" "$args4" "$args5" "$args6" "$args7" ;;
(9) set -- "$args0" "$args1" "$args2" "$args3" "$args4" "$args5" "$args6" "$args7" "$args8" ;;
esac
fi
# Split up the JVM_OPTS And GRADLE_OPTS values into an array, following the shell quoting and substitution rules
function splitJvmOpts() {
JVM_OPTS=("$@")
}
eval splitJvmOpts $DEFAULT_JVM_OPTS $JAVA_OPTS $GRADLE_OPTS
JVM_OPTS[${#JVM_OPTS[*]}]="-Dorg.gradle.appname=$APP_BASE_NAME"
exec "$JAVACMD" "${JVM_OPTS[@]}" -classpath "$CLASSPATH" org.gradle.wrapper.GradleWrapperMain "$@"

90
gradlew.bat vendored Normal file
View file

@ -0,0 +1,90 @@
@if "%DEBUG%" == "" @echo off
@rem ##########################################################################
@rem
@rem Gradle startup script for Windows
@rem
@rem ##########################################################################
@rem Set local scope for the variables with windows NT shell
if "%OS%"=="Windows_NT" setlocal
@rem Add default JVM options here. You can also use JAVA_OPTS and GRADLE_OPTS to pass JVM options to this script.
set DEFAULT_JVM_OPTS=
set DIRNAME=%~dp0
if "%DIRNAME%" == "" set DIRNAME=.
set APP_BASE_NAME=%~n0
set APP_HOME=%DIRNAME%
@rem Find java.exe
if defined JAVA_HOME goto findJavaFromJavaHome
set JAVA_EXE=java.exe
%JAVA_EXE% -version >NUL 2>&1
if "%ERRORLEVEL%" == "0" goto init
echo.
echo ERROR: JAVA_HOME is not set and no 'java' command could be found in your PATH.
echo.
echo Please set the JAVA_HOME variable in your environment to match the
echo location of your Java installation.
goto fail
:findJavaFromJavaHome
set JAVA_HOME=%JAVA_HOME:"=%
set JAVA_EXE=%JAVA_HOME%/bin/java.exe
if exist "%JAVA_EXE%" goto init
echo.
echo ERROR: JAVA_HOME is set to an invalid directory: %JAVA_HOME%
echo.
echo Please set the JAVA_HOME variable in your environment to match the
echo location of your Java installation.
goto fail
:init
@rem Get command-line arguments, handling Windowz variants
if not "%OS%" == "Windows_NT" goto win9xME_args
if "%@eval[2+2]" == "4" goto 4NT_args
:win9xME_args
@rem Slurp the command line arguments.
set CMD_LINE_ARGS=
set _SKIP=2
:win9xME_args_slurp
if "x%~1" == "x" goto execute
set CMD_LINE_ARGS=%*
goto execute
:4NT_args
@rem Get arguments from the 4NT Shell from JP Software
set CMD_LINE_ARGS=%$
:execute
@rem Setup the command line
set CLASSPATH=%APP_HOME%\gradle\wrapper\gradle-wrapper.jar
@rem Execute Gradle
"%JAVA_EXE%" %DEFAULT_JVM_OPTS% %JAVA_OPTS% %GRADLE_OPTS% "-Dorg.gradle.appname=%APP_BASE_NAME%" -classpath "%CLASSPATH%" org.gradle.wrapper.GradleWrapperMain %CMD_LINE_ARGS%
:end
@rem End local scope for the variables with windows NT shell
if "%ERRORLEVEL%"=="0" goto mainEnd
:fail
rem Set variable GRADLE_EXIT_CONSOLE if you need the _script_ return code instead of
rem the _cmd.exe /c_ return code!
if not "" == "%GRADLE_EXIT_CONSOLE%" exit 1
exit /b 1
:mainEnd
if "%OS%"=="Windows_NT" endlocal
:omega

Binary file not shown.

Binary file not shown.

Binary file not shown.

Binary file not shown.

Binary file not shown.

1
settings.gradle Normal file
View file

@ -0,0 +1 @@
include ':app'

View file

@ -1,705 +0,0 @@
/*
* @(#)Emaily.java
*
* Copyright (c) 2011-2012 Erik C. Thauvin (http://erik.thauvin.net/)
* All rights reserved.
*
* Redistribution and use in source and binary forms, with or without
* modification, are permitted provided that the following conditions are
* met:
*
* Redistributions of source code must retain the above copyright notice,
* this list of conditions and the following disclaimer.
*
* Redistributions in binary form must reproduce the above copyright notice,
* this list of conditions and the following disclaimer in the documentation
* and/or other materials provided with the distribution.
*
* Neither the name of the authors nor the names of its contributors may be
* used to endorse or promote products derived from this software without
* specific prior written permission.
*
* THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS
* IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO,
* THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR
* PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT OWNER OR
* CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL,
* EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO,
* PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR
* PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF
* LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING
* NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS
* SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
*
* $Id$
*
*/
package net.thauvin.erik.android.emaily;
import static com.rosaloves.bitlyj.Bitly.as;
import static com.rosaloves.bitlyj.Bitly.shorten;
import java.io.IOException;
import java.net.MalformedURLException;
import java.net.URL;
import java.net.UnknownHostException;
import java.util.Date;
import android.accounts.Account;
import android.accounts.AccountManager;
import android.accounts.AccountManagerCallback;
import android.accounts.AccountManagerFuture;
import android.accounts.OperationCanceledException;
import android.app.Activity;
import android.app.AlertDialog;
import android.app.ProgressDialog;
import android.content.Context;
import android.content.DialogInterface;
import android.content.Intent;
import android.content.SharedPreferences;
import android.content.SharedPreferences.Editor;
import android.content.pm.PackageManager.NameNotFoundException;
import android.os.AsyncTask;
import android.os.Bundle;
import android.preference.PreferenceManager;
import android.text.ClipboardManager;
import android.text.Html;
import android.util.Log;
import android.widget.Toast;
import com.google.api.client.googleapis.extensions.android2.auth.GoogleAccountManager;
import com.google.api.client.googleapis.json.GoogleJsonError;
import com.google.api.client.googleapis.json.GoogleJsonResponseException;
import com.google.api.client.http.HttpTransport;
import com.google.api.client.http.javanet.NetHttpTransport;
import com.google.api.client.http.json.JsonHttpRequest;
import com.google.api.client.http.json.JsonHttpRequestInitializer;
import com.google.api.client.json.JsonFactory;
import com.google.api.client.json.jackson.JacksonFactory;
import com.google.api.services.urlshortener.Urlshortener;
import com.google.api.services.urlshortener.UrlshortenerRequest;
import com.google.api.services.urlshortener.model.Url;
/**
* The <code>Emaily</code> class implements a URL shortener intent.
*
* @author <a href="mailto:erik@thauvin.net">Erik C. Thauvin</a>
* @version $Revision$
* @created Oct 11, 2011
* @since 1.0
*/
@SuppressWarnings("deprecation")
public class Emaily extends Activity
{
private static final String ACCOUNT_TYPE = "com.google";
private static final String OAUTH_URL = "oauth2:https://www.googleapis.com/auth/urlshortener";
private String appName;
private SharedPreferences sharedPrefs;
@Override
public void onCreate(Bundle savedInstanceState)
{
super.onCreate(savedInstanceState);
final Intent intent = getIntent();
appName = getString(R.string.app_name);
sharedPrefs = PreferenceManager.getDefaultSharedPreferences(this);
if (Intent.ACTION_SEND.equals(intent.getAction()))
{
final boolean isGoogl = getBoolPref(R.string.prefs_key_googl_enabled, true);
if (isGoogl)
{
final String account = getPref(R.string.prefs_key_googl_account);
if (isValid(account))
{
startEmailyTask(intent, new Account(account, ACCOUNT_TYPE), false);
}
else
{
final AlertDialog.Builder builder = new AlertDialog.Builder(this);
builder.setTitle(R.string.dialog_accounts_title);
final Account[] accounts = AccountManager.get(this).getAccountsByType(ACCOUNT_TYPE);
final int size = accounts.length;
if (size > 0)
{
if (size == 1)
{
startEmailyTask(intent, accounts[0], false);
}
else
{
final CharSequence[] names = new CharSequence[size];
for (int i = 0; i < size; i++)
{
names[i] = accounts[i].name;
}
builder.setSingleChoiceItems(names, 0, new DialogInterface.OnClickListener()
{
@Override
public void onClick(DialogInterface dialog, int which)
{
dialog.dismiss();
final Editor editor = sharedPrefs.edit();
editor.putString(getString(R.string.prefs_key_googl_account), names[which].toString());
editor.putLong(getString(R.string.prefs_key_googl_token_expiry), 0L);
editor.commit();
startEmailyTask(intent, accounts[which], false);
}
});
builder.create().show();
}
}
else
{
startEmailyTask(intent, isGoogl, false);
}
}
}
else
{
startEmailyTask(intent, isGoogl, false);
}
}
else
{
Emaily.this.finish();
}
}
/**
* Starts the task.
*
* @param intent The original intent.
* @param sharedPrefs The shared preference.
* @param isGoogl The goo.gl flag.
* @param isRetry The retry flag.
*/
private void startEmailyTask(final Intent intent, final boolean isGoogl, final boolean isRetry)
{
final EmailyTask task;
if (isGoogl)
{
task = new EmailyTask(getPref(R.string.prefs_key_googl_account), getPref(R.string.prefs_key_googl_token), isGoogl,
getBoolPref(R.string.prefs_key_html_chkbox), isRetry);
}
else
{
task = new EmailyTask(getPref(R.string.prefs_key_bitly_username), getPref(R.string.prefs_key_bitly_apikey), isGoogl,
getBoolPref(R.string.prefs_key_html_chkbox), isRetry);
}
task.execute(intent);
}
/**
* Starts the task.
*
* @param intent The original intent.
* @param account The account.
* @param isRetry The retry flag.
*/
private void startEmailyTask(final Intent intent, final Account account, final boolean isRetry)
{
final GoogleAccountManager googleAccountManager = new GoogleAccountManager(Emaily.this);
final long expiry = sharedPrefs.getLong(getString(R.string.prefs_key_googl_token_expiry), 0L);
final long now = System.currentTimeMillis();
final long maxLife = (60L * 55L) * 1000L; // 55 minutes
Log.d(appName, "Token Expires: " + new Date(expiry));
if (expiry >= (now + maxLife) || expiry <= now)
{
final String token = getPref(R.string.prefs_key_googl_token);
if (isValid(token))
{
googleAccountManager.manager.invalidateAuthToken(ACCOUNT_TYPE, token);
Log.d(appName, "Token Invalidated: " + token);
}
}
googleAccountManager.manager.getAuthToken(account, OAUTH_URL, null, Emaily.this, new AccountManagerCallback<Bundle>()
{
@Override
public void run(AccountManagerFuture<Bundle> future)
{
try
{
final String token = future.getResult().getString(AccountManager.KEY_AUTHTOKEN);
final Editor editor = sharedPrefs.edit();
final long now = System.currentTimeMillis();
final long expires;
if (expiry < now)
{
expires = now + maxLife;
}
else
{
expires = expiry;
}
editor.putLong(getString(R.string.prefs_key_googl_token_expiry), expires);
editor.putString(getString(R.string.prefs_key_googl_token), token);
editor.commit();
Log.d(appName, account.toString());
Log.d(appName, "Token: " + token);
Log.d(appName, "Expires: " + new Date(expires));
startEmailyTask(intent, true, isRetry);
}
catch (OperationCanceledException e)
{
Log.e(appName, "Auth token request has been canceled.", e);
}
catch (Exception e)
{
Log.e(appName, "Exception while requesting the auth token.", e);
}
}
}, null);
}
/**
* Retries the task.
*
* @param intent The original intent.
*/
public void retry(final Intent intent)
{
sharedPrefs.edit().putLong(getString(R.string.prefs_key_googl_token_expiry), 0L).commit();
startEmailyTask(intent, new Account(getPref(R.string.prefs_key_googl_account), ACCOUNT_TYPE), true);
}
/**
* Validates a string.
*
* @param s The string to validate.
* @return returns <code>true</code> if the string is not empty or null, <code>false</code> otherwise.
*/
public static boolean isValid(String s)
{
return (s != null) && (!s.trim().isEmpty());
}
/**
* Returns the value of the specified shared reference based on the specified string id. The default value is an empty string.
*
* @param sharedPrefs The shared preference.
* @param id The string id.
* @return The preference value.
*/
public String getPref(int id)
{
return getPref(id, "");
}
/**
* Returns the value of the specified shared reference based on the specified string id.
*
* @param sharedPrefs The shared preference.
* @param id The string id.
* @param defaultValue The default value, used if the preference is empty.
* @return The preference value.
*/
public String getPref(int id, String defaultValue)
{
return sharedPrefs.getString(getString(id), defaultValue);
}
/**
* Returns the value of the specified shared reference based on the specified string id. The default value is <code>false</code>.
*
* @param sharedPrefs The shared preference.
* @param id The string id.
* @return The preference value.
*/
public boolean getBoolPref(int id)
{
return getBoolPref(id, false);
}
/**
* Returns the value of the specified shared reference based on the specified string id.
*
* @param sharedPrefs The shared preference.
* @param id The string id.
* @param defaultValue The default value, used if the preference is empty.
* @return The preference value.
*/
public boolean getBoolPref(int id, boolean defaultValue)
{
return sharedPrefs.getBoolean(getString(id), defaultValue);
}
/**
* The <code>EmailyTask</code> class.
*/
private class EmailyTask extends AsyncTask<Intent, Void, EmailyResult>
{
private final ProgressDialog dialog = new ProgressDialog(Emaily.this);
private final String username;
private final String keytoken;
private final boolean isGoogl;
private final boolean isHtml;
private final boolean isRetry;
public EmailyTask(String username, String keytoken, boolean isGoogl, boolean isHtml, boolean isRetry)
{
this.username = username;
this.keytoken = keytoken;
this.isGoogl = isGoogl;
this.isHtml = isHtml;
this.isRetry = isRetry;
}
@Override
protected EmailyResult doInBackground(Intent... intent)
{
final EmailyResult result = new EmailyResult(intent[0]);
final Intent emailIntent = new Intent(Intent.ACTION_SEND);
if (isHtml)
{
emailIntent.setType("text/html");
}
else
{
emailIntent.setType("text/plain");
}
final Bundle extras = intent[0].getExtras();
final String pageUrl = extras.getString(Intent.EXTRA_TEXT);
final String pageTitle = extras.getString(Intent.EXTRA_SUBJECT);
final StringBuilder textBefore = new StringBuilder();
if (isValid(pageTitle))
{
emailIntent.putExtra(Intent.EXTRA_SUBJECT, pageTitle);
}
final boolean hasCredentials = isValid(username) && isValid(keytoken);
final StringBuilder shortUrl = new StringBuilder();
if (isValid(pageUrl))
{
final HttpTransport transport = new NetHttpTransport();
final JsonFactory jsonFactory = new JacksonFactory();
String version = "";
try
{
version = '/' + getPackageManager().getPackageInfo(getPackageName(), 0).versionName;
}
catch (NameNotFoundException ignore)
{
// Do nothing;
}
final Url toInsert = new Url();
final String[] splits = pageUrl.split("\\s");
for (String item : splits)
{
try
{
new URL(item.trim());
if (isGoogl || !hasCredentials)
{
Log.d(appName, "goo.gl -> " + item);
final Urlshortener shortener = com.google.api.services.urlshortener.Urlshortener
.builder(transport, jsonFactory).setApplicationName(appName + version)
.setJsonHttpRequestInitializer(new JsonHttpRequestInitializer()
{
@Override
public void initialize(JsonHttpRequest request) throws IOException
{
UrlshortenerRequest shortnerRequest = (UrlshortenerRequest) request;
shortnerRequest.setKey(getString(R.string.secret_apikey));
if (isValid(keytoken))
{
shortnerRequest.setOauthToken(keytoken);
}
shortnerRequest.put("client_id", getString(R.string.secret_client_id));
shortnerRequest.put("client_secret", getString(R.string.secret_client_secret));
}
}).build();
toInsert.setLongUrl(item.trim());
try
{
final Url shortened = shortener.url().insert(toInsert).execute();
shortUrl.append(shortened.getId());
}
catch (GoogleJsonResponseException e)
{
result.setCode(R.string.alert_error);
final GoogleJsonError err = e.getDetails();
result.setMessage(err.message);
if (err.code == 401)
{
if (!isRetry)
{
result.setRetry(true);
}
}
Log.e(appName, "Exception while shortening '" + item + "' via goo.gl.", e);
}
catch (UnknownHostException e)
{
result.setCode(R.string.alert_nohost);
result.setMessage(e.getMessage());
Log.e(appName, "UnknownHostException while shortening '" + item + "' via goo.gl.", e);
}
catch (IOException e)
{
result.setCode(R.string.alert_error);
result.setMessage(e.getMessage());
Log.e(appName, "IOException while shortening '" + item + "' via goo.gl.", e);
}
}
else
{
Log.d(appName, "bit.ly -> " + item);
try
{
shortUrl.append(as(username, keytoken).call(shorten(item.trim())).getShortUrl());
}
catch (Exception e)
{
final Throwable cause = e.getCause();
if (cause != null && cause instanceof UnknownHostException)
{
result.setCode(R.string.alert_nohost);
result.setMessage(cause.getMessage());
}
else
{
result.setCode(R.string.alert_error);
result.setMessage(e.getMessage());
}
Log.e(appName, "Exception while shortening '" + item + "' via bit.ly.", e);
}
break;
}
break;
}
catch (MalformedURLException mue)
{
Log.d(appName, "Attempted to process an invalid URL: " + item, mue);
if (textBefore.length() > 0)
{
textBefore.append(" ");
}
textBefore.append(item);
}
}
}
else
{
result.setCode(R.string.alert_nocreds);
}
if (!result.isRetry())
{
if (shortUrl.length() > 0)
{
if (isHtml)
{
emailIntent.putExtra(Intent.EXTRA_TEXT, Html.fromHtml("<a href=\"" + shortUrl + "\">" + shortUrl + "</a>"));
}
else
{
emailIntent.putExtra(Intent.EXTRA_TEXT, shortUrl.toString());
}
if (!isValid(pageTitle) && textBefore.length() > 0)
{
emailIntent.putExtra(Intent.EXTRA_SUBJECT, textBefore.toString());
}
}
else
{
final CharSequence chars = extras.getCharSequence(Intent.EXTRA_TEXT);
if (chars.length() > 0)
{
emailIntent.putExtra(Intent.EXTRA_TEXT, chars);
}
else if (isValid(pageUrl))
{
emailIntent.putExtra(Intent.EXTRA_TEXT, pageUrl);
}
}
try
{
startActivity(emailIntent);
}
catch (android.content.ActivityNotFoundException ignore)
{
if (!result.hasError() && shortUrl.length() > 0)
{
final ClipboardManager clip = (ClipboardManager) getSystemService(Context.CLIPBOARD_SERVICE);
clip.setText(shortUrl);
result.setCode(R.string.alert_notfound_clip);
}
else
{
result.setCode(R.string.alert_notfound);
}
}
}
return result;
}
@Override
protected void onPostExecute(EmailyResult result)
{
if (this.dialog.isShowing())
{
this.dialog.dismiss();
}
if (result.isRetry())
{
Emaily.this.retry(result.getIntent());
}
else
{
if (result.hasError())
{
Toast.makeText(
getApplicationContext(),
getString(result.getCode(), result.getMessage(), isGoogl ? getString(R.string.prefs_googl_title)
: getString(R.string.prefs_bitly_title)), Toast.LENGTH_LONG).show();
}
Emaily.this.finish();
}
}
@Override
protected void onPreExecute()
{
if (isRetry)
{
this.dialog.setMessage(getString(R.string.progress_msg_retry));
}
else
{
this.dialog.setMessage(getString(R.string.progress_msg));
}
this.dialog.show();
}
}
/**
* The <code>EmailyResult</code> class.
*/
private class EmailyResult
{
private int code = 0;
private String message;
private boolean retry = false;
private final Intent intent;
public EmailyResult(Intent intent)
{
this.intent = intent;
}
public int getCode()
{
return code;
}
public String getMessage()
{
if (isValid(message))
{
return message;
}
else
{
return "";
}
}
public Intent getIntent()
{
return intent;
}
public boolean hasError()
{
return code != 0;
}
public boolean isRetry()
{
return retry;
}
public void setCode(int code)
{
this.code = code;
}
public void setMessage(String message)
{
this.message = message;
}
public void setRetry(boolean retry)
{
this.retry = retry;
}
}
}