diff --git a/assets/1f601.png b/assets/1f601.png
new file mode 100644
index 0000000000..591cfcef8b
Binary files /dev/null and b/assets/1f601.png differ
diff --git a/assets/1f602.png b/assets/1f602.png
new file mode 100644
index 0000000000..47df693d42
Binary files /dev/null and b/assets/1f602.png differ
diff --git a/assets/1f603.png b/assets/1f603.png
new file mode 100644
index 0000000000..77b581d68f
Binary files /dev/null and b/assets/1f603.png differ
diff --git a/assets/1f604.png b/assets/1f604.png
new file mode 100644
index 0000000000..81a8396899
Binary files /dev/null and b/assets/1f604.png differ
diff --git a/assets/1f605.png b/assets/1f605.png
new file mode 100644
index 0000000000..3903f717f3
Binary files /dev/null and b/assets/1f605.png differ
diff --git a/assets/1f606.png b/assets/1f606.png
new file mode 100644
index 0000000000..11c91eb22e
Binary files /dev/null and b/assets/1f606.png differ
diff --git a/assets/1f609.png b/assets/1f609.png
new file mode 100644
index 0000000000..756766dd3e
Binary files /dev/null and b/assets/1f609.png differ
diff --git a/assets/1f60a.png b/assets/1f60a.png
new file mode 100644
index 0000000000..1e9021cb6f
Binary files /dev/null and b/assets/1f60a.png differ
diff --git a/assets/1f60b.png b/assets/1f60b.png
new file mode 100644
index 0000000000..fc39637ecd
Binary files /dev/null and b/assets/1f60b.png differ
diff --git a/assets/1f60c.png b/assets/1f60c.png
new file mode 100644
index 0000000000..820cf315a1
Binary files /dev/null and b/assets/1f60c.png differ
diff --git a/assets/1f60d.png b/assets/1f60d.png
new file mode 100644
index 0000000000..0e5794270e
Binary files /dev/null and b/assets/1f60d.png differ
diff --git a/assets/1f60f.png b/assets/1f60f.png
new file mode 100644
index 0000000000..bc6e5082c8
Binary files /dev/null and b/assets/1f60f.png differ
diff --git a/assets/1f612.png b/assets/1f612.png
new file mode 100644
index 0000000000..3722e6f575
Binary files /dev/null and b/assets/1f612.png differ
diff --git a/assets/1f613.png b/assets/1f613.png
new file mode 100644
index 0000000000..e894b76996
Binary files /dev/null and b/assets/1f613.png differ
diff --git a/assets/1f614.png b/assets/1f614.png
new file mode 100644
index 0000000000..2f3bad9453
Binary files /dev/null and b/assets/1f614.png differ
diff --git a/assets/1f616.png b/assets/1f616.png
new file mode 100644
index 0000000000..a5877a0a79
Binary files /dev/null and b/assets/1f616.png differ
diff --git a/assets/1f618.png b/assets/1f618.png
new file mode 100644
index 0000000000..af9a80b7f0
Binary files /dev/null and b/assets/1f618.png differ
diff --git a/assets/1f61a.png b/assets/1f61a.png
new file mode 100644
index 0000000000..449de19704
Binary files /dev/null and b/assets/1f61a.png differ
diff --git a/assets/1f61c.png b/assets/1f61c.png
new file mode 100644
index 0000000000..6ae9d497d3
Binary files /dev/null and b/assets/1f61c.png differ
diff --git a/assets/1f61d.png b/assets/1f61d.png
new file mode 100644
index 0000000000..333716ee1f
Binary files /dev/null and b/assets/1f61d.png differ
diff --git a/assets/1f61e.png b/assets/1f61e.png
new file mode 100644
index 0000000000..8255200871
Binary files /dev/null and b/assets/1f61e.png differ
diff --git a/assets/1f620.png b/assets/1f620.png
new file mode 100644
index 0000000000..34174f5e5c
Binary files /dev/null and b/assets/1f620.png differ
diff --git a/assets/1f621.png b/assets/1f621.png
new file mode 100644
index 0000000000..c65ddff552
Binary files /dev/null and b/assets/1f621.png differ
diff --git a/assets/1f622.png b/assets/1f622.png
new file mode 100644
index 0000000000..6d0d9afd28
Binary files /dev/null and b/assets/1f622.png differ
diff --git a/assets/1f623.png b/assets/1f623.png
new file mode 100644
index 0000000000..c7e433e8ec
Binary files /dev/null and b/assets/1f623.png differ
diff --git a/assets/1f624.png b/assets/1f624.png
new file mode 100644
index 0000000000..92f93bd102
Binary files /dev/null and b/assets/1f624.png differ
diff --git a/assets/1f625.png b/assets/1f625.png
new file mode 100644
index 0000000000..fa5f9e7f9f
Binary files /dev/null and b/assets/1f625.png differ
diff --git a/assets/1f628.png b/assets/1f628.png
new file mode 100644
index 0000000000..513fce47b6
Binary files /dev/null and b/assets/1f628.png differ
diff --git a/assets/1f629.png b/assets/1f629.png
new file mode 100644
index 0000000000..0c5475411c
Binary files /dev/null and b/assets/1f629.png differ
diff --git a/assets/1f62a.png b/assets/1f62a.png
new file mode 100644
index 0000000000..df4f55efd9
Binary files /dev/null and b/assets/1f62a.png differ
diff --git a/assets/1f62b.png b/assets/1f62b.png
new file mode 100644
index 0000000000..3a8eefe565
Binary files /dev/null and b/assets/1f62b.png differ
diff --git a/assets/1f62d.png b/assets/1f62d.png
new file mode 100644
index 0000000000..7d433183aa
Binary files /dev/null and b/assets/1f62d.png differ
diff --git a/assets/1f630.png b/assets/1f630.png
new file mode 100644
index 0000000000..b9e39bc60f
Binary files /dev/null and b/assets/1f630.png differ
diff --git a/assets/1f631.png b/assets/1f631.png
new file mode 100644
index 0000000000..76bfc6b8a6
Binary files /dev/null and b/assets/1f631.png differ
diff --git a/assets/1f632.png b/assets/1f632.png
new file mode 100644
index 0000000000..858a83484a
Binary files /dev/null and b/assets/1f632.png differ
diff --git a/assets/1f633.png b/assets/1f633.png
new file mode 100644
index 0000000000..9b49410c0c
Binary files /dev/null and b/assets/1f633.png differ
diff --git a/assets/1f635.png b/assets/1f635.png
new file mode 100644
index 0000000000..8001d6ff8f
Binary files /dev/null and b/assets/1f635.png differ
diff --git a/assets/1f637.png b/assets/1f637.png
new file mode 100644
index 0000000000..05887e99c6
Binary files /dev/null and b/assets/1f637.png differ
diff --git a/assets/1f638.png b/assets/1f638.png
new file mode 100644
index 0000000000..ad333ba3b6
Binary files /dev/null and b/assets/1f638.png differ
diff --git a/assets/1f639.png b/assets/1f639.png
new file mode 100644
index 0000000000..6c60cb0efc
Binary files /dev/null and b/assets/1f639.png differ
diff --git a/assets/1f63a.png b/assets/1f63a.png
new file mode 100644
index 0000000000..dbf1b0276a
Binary files /dev/null and b/assets/1f63a.png differ
diff --git a/assets/1f63b.png b/assets/1f63b.png
new file mode 100644
index 0000000000..eeba240e53
Binary files /dev/null and b/assets/1f63b.png differ
diff --git a/assets/1f63c.png b/assets/1f63c.png
new file mode 100644
index 0000000000..351565e246
Binary files /dev/null and b/assets/1f63c.png differ
diff --git a/assets/1f63d.png b/assets/1f63d.png
new file mode 100644
index 0000000000..adc62fbe3c
Binary files /dev/null and b/assets/1f63d.png differ
diff --git a/assets/1f63e.png b/assets/1f63e.png
new file mode 100644
index 0000000000..4325fd48dd
Binary files /dev/null and b/assets/1f63e.png differ
diff --git a/assets/1f63f.png b/assets/1f63f.png
new file mode 100644
index 0000000000..42d4c27cab
Binary files /dev/null and b/assets/1f63f.png differ
diff --git a/assets/1f640.png b/assets/1f640.png
new file mode 100644
index 0000000000..d94cd34ff5
Binary files /dev/null and b/assets/1f640.png differ
diff --git a/assets/1f645.png b/assets/1f645.png
new file mode 100644
index 0000000000..d459a35bc1
Binary files /dev/null and b/assets/1f645.png differ
diff --git a/assets/1f646.png b/assets/1f646.png
new file mode 100644
index 0000000000..e8b98194ed
Binary files /dev/null and b/assets/1f646.png differ
diff --git a/assets/1f647.png b/assets/1f647.png
new file mode 100644
index 0000000000..024cb61049
Binary files /dev/null and b/assets/1f647.png differ
diff --git a/assets/1f648.png b/assets/1f648.png
new file mode 100644
index 0000000000..0890a62227
Binary files /dev/null and b/assets/1f648.png differ
diff --git a/assets/1f649.png b/assets/1f649.png
new file mode 100644
index 0000000000..f97a1f9a09
Binary files /dev/null and b/assets/1f649.png differ
diff --git a/assets/1f64a.png b/assets/1f64a.png
new file mode 100644
index 0000000000..87944c4de5
Binary files /dev/null and b/assets/1f64a.png differ
diff --git a/assets/1f64b.png b/assets/1f64b.png
new file mode 100644
index 0000000000..e1741a40e7
Binary files /dev/null and b/assets/1f64b.png differ
diff --git a/assets/1f64c.png b/assets/1f64c.png
new file mode 100644
index 0000000000..e03142bdce
Binary files /dev/null and b/assets/1f64c.png differ
diff --git a/assets/1f64d.png b/assets/1f64d.png
new file mode 100644
index 0000000000..6f34d5e159
Binary files /dev/null and b/assets/1f64d.png differ
diff --git a/assets/1f64e.png b/assets/1f64e.png
new file mode 100644
index 0000000000..c4a95c3b2a
Binary files /dev/null and b/assets/1f64e.png differ
diff --git a/assets/1f64f.png b/assets/1f64f.png
new file mode 100644
index 0000000000..f86c992d5a
Binary files /dev/null and b/assets/1f64f.png differ
diff --git a/res/drawable-hdpi/ic_emoji_dark.png b/res/drawable-hdpi/ic_emoji_dark.png
new file mode 100644
index 0000000000..b82c965395
Binary files /dev/null and b/res/drawable-hdpi/ic_emoji_dark.png differ
diff --git a/res/drawable-hdpi/ic_emoji_light.png b/res/drawable-hdpi/ic_emoji_light.png
new file mode 100644
index 0000000000..c6250598e0
Binary files /dev/null and b/res/drawable-hdpi/ic_emoji_light.png differ
diff --git a/res/drawable-hdpi/ic_emoji_recent_light.png b/res/drawable-hdpi/ic_emoji_recent_light.png
new file mode 100644
index 0000000000..b28a4ee86c
Binary files /dev/null and b/res/drawable-hdpi/ic_emoji_recent_light.png differ
diff --git a/res/drawable-hdpi/ic_ime_dark.png b/res/drawable-hdpi/ic_ime_dark.png
new file mode 100644
index 0000000000..c2dc4d3804
Binary files /dev/null and b/res/drawable-hdpi/ic_ime_dark.png differ
diff --git a/res/drawable-hdpi/ic_ime_light.png b/res/drawable-hdpi/ic_ime_light.png
new file mode 100644
index 0000000000..14d7a5eba4
Binary files /dev/null and b/res/drawable-hdpi/ic_ime_light.png differ
diff --git a/res/drawable-mdpi/ic_emoji_dark.png b/res/drawable-mdpi/ic_emoji_dark.png
new file mode 100644
index 0000000000..d99344a4de
Binary files /dev/null and b/res/drawable-mdpi/ic_emoji_dark.png differ
diff --git a/res/drawable-mdpi/ic_emoji_light.png b/res/drawable-mdpi/ic_emoji_light.png
new file mode 100644
index 0000000000..5d34a9e146
Binary files /dev/null and b/res/drawable-mdpi/ic_emoji_light.png differ
diff --git a/res/drawable-mdpi/ic_emoji_recent_light.png b/res/drawable-mdpi/ic_emoji_recent_light.png
new file mode 100644
index 0000000000..8d6cd6a40d
Binary files /dev/null and b/res/drawable-mdpi/ic_emoji_recent_light.png differ
diff --git a/res/drawable-mdpi/ic_ime_dark.png b/res/drawable-mdpi/ic_ime_dark.png
new file mode 100644
index 0000000000..3145b74c62
Binary files /dev/null and b/res/drawable-mdpi/ic_ime_dark.png differ
diff --git a/res/drawable-mdpi/ic_ime_light.png b/res/drawable-mdpi/ic_ime_light.png
new file mode 100644
index 0000000000..f749583df2
Binary files /dev/null and b/res/drawable-mdpi/ic_ime_light.png differ
diff --git a/res/drawable-xhdpi/ic_emoji_dark.png b/res/drawable-xhdpi/ic_emoji_dark.png
new file mode 100644
index 0000000000..bfa252c286
Binary files /dev/null and b/res/drawable-xhdpi/ic_emoji_dark.png differ
diff --git a/res/drawable-xhdpi/ic_emoji_light.png b/res/drawable-xhdpi/ic_emoji_light.png
new file mode 100644
index 0000000000..bc2146d2fa
Binary files /dev/null and b/res/drawable-xhdpi/ic_emoji_light.png differ
diff --git a/res/drawable-xhdpi/ic_emoji_recent_light.png b/res/drawable-xhdpi/ic_emoji_recent_light.png
new file mode 100644
index 0000000000..a3d9ca3684
Binary files /dev/null and b/res/drawable-xhdpi/ic_emoji_recent_light.png differ
diff --git a/res/drawable-xhdpi/ic_ime_dark.png b/res/drawable-xhdpi/ic_ime_dark.png
new file mode 100644
index 0000000000..2de719c32e
Binary files /dev/null and b/res/drawable-xhdpi/ic_ime_dark.png differ
diff --git a/res/drawable-xhdpi/ic_ime_light.png b/res/drawable-xhdpi/ic_ime_light.png
new file mode 100644
index 0000000000..0ff5dd8604
Binary files /dev/null and b/res/drawable-xhdpi/ic_ime_light.png differ
diff --git a/res/drawable/emoji_toggle_background.xml b/res/drawable/emoji_toggle_background.xml
new file mode 100644
index 0000000000..844ebeadae
--- /dev/null
+++ b/res/drawable/emoji_toggle_background.xml
@@ -0,0 +1,13 @@
+
+
+
+
+
+
+
+
+
+
+
\ No newline at end of file
diff --git a/res/layout/conversation_activity.xml b/res/layout/conversation_activity.xml
index d4bcac9f6b..fcfdd132f0 100644
--- a/res/layout/conversation_activity.xml
+++ b/res/layout/conversation_activity.xml
@@ -1,90 +1,97 @@
-
-
+
+ android:id="@+id/layout_container"
+ android:layout_width="fill_parent"
+ android:layout_height="fill_parent"
+ android:layout_weight="1"
+ android:orientation="vertical"
+ android:background="?conversation_background"
+ android:gravity="bottom">
+ android:id="@+id/fragment_content"
+ android:name="org.thoughtcrime.securesms.ConversationFragment"
+ android:layout_width="match_parent"
+ android:layout_height="match_parent"
+ android:layout_above="@+id/bottom_container" />
-
-
-
-
-
-
+
-
-
-
-
-
-
+ android:id="@+id/attachment_editor"
+ android:layout_width="fill_parent"
+ android:layout_height="wrap_content"
+ android:orientation="horizontal"
+ android:visibility="gone">
+
+
+
+
+
+
+
+
+
+
-
-
+
+
-
-
-
-
-
+
+
+
-
+
+
+
+
+
+
+
+
diff --git a/res/layout/emoji_drawer.xml b/res/layout/emoji_drawer.xml
new file mode 100644
index 0000000000..ee12dbb0ad
--- /dev/null
+++ b/res/layout/emoji_drawer.xml
@@ -0,0 +1,30 @@
+
+
+
+
+
+
+
+
+
+
+
\ No newline at end of file
diff --git a/res/layout/emoji_grid_layout.xml b/res/layout/emoji_grid_layout.xml
new file mode 100644
index 0000000000..958cc24422
--- /dev/null
+++ b/res/layout/emoji_grid_layout.xml
@@ -0,0 +1,20 @@
+
+
+
+
+
+
+
\ No newline at end of file
diff --git a/res/values/attrs.xml b/res/values/attrs.xml
index b662e79d12..7dca2176fd 100644
--- a/res/values/attrs.xml
+++ b/res/values/attrs.xml
@@ -16,6 +16,8 @@
+
+
diff --git a/res/values/themes.xml b/res/values/themes.xml
index e406496bd9..ecea596452 100644
--- a/res/values/themes.xml
+++ b/res/values/themes.xml
@@ -17,6 +17,8 @@
- @drawable/ic_send_holo_light
- @drawable/ic_send_encrypted_holo_light
- @drawable/ic_sms_mms_delivered_light
+ - @drawable/ic_emoji_dark
+ - @drawable/ic_ime_dark
diff --git a/src/org/thoughtcrime/securesms/ConversationActivity.java b/src/org/thoughtcrime/securesms/ConversationActivity.java
index f4c22c66c0..a873f8a455 100644
--- a/src/org/thoughtcrime/securesms/ConversationActivity.java
+++ b/src/org/thoughtcrime/securesms/ConversationActivity.java
@@ -41,6 +41,7 @@ import android.view.View.OnClickListener;
import android.view.View.OnKeyListener;
import android.view.WindowManager;
import android.view.inputmethod.EditorInfo;
+import android.view.inputmethod.InputMethodManager;
import android.widget.EditText;
import android.widget.ImageButton;
import android.widget.TextView;
@@ -49,6 +50,8 @@ import android.widget.Toast;
import com.actionbarsherlock.view.Menu;
import com.actionbarsherlock.view.MenuInflater;
import com.actionbarsherlock.view.MenuItem;
+import org.thoughtcrime.securesms.components.EmojiDrawer;
+import org.thoughtcrime.securesms.components.EmojiToggle;
import org.thoughtcrime.securesms.components.RecipientsPanel;
import org.thoughtcrime.securesms.crypto.KeyExchangeInitiator;
import org.thoughtcrime.securesms.crypto.KeyExchangeProcessor;
@@ -122,6 +125,8 @@ public class ConversationActivity extends PassphraseRequiredSherlockFragmentActi
private AttachmentTypeSelectorAdapter attachmentAdapter;
private AttachmentManager attachmentManager;
private BroadcastReceiver securityUpdateReceiver;
+ private EmojiDrawer emojiDrawer;
+ private EmojiToggle emojiToggle;
private Recipients recipients;
private long threadId;
@@ -282,6 +287,16 @@ public class ConversationActivity extends PassphraseRequiredSherlockFragmentActi
return false;
}
+ @Override
+ public void onBackPressed() {
+ if (emojiDrawer.getVisibility() == View.VISIBLE) {
+ emojiDrawer.setVisibility(View.GONE);
+ emojiToggle.toggle();
+ } else {
+ super.onBackPressed();
+ }
+ }
+
//////// Event Handlers
private void handleReturnToConversationList() {
@@ -566,19 +581,25 @@ public class ConversationActivity extends PassphraseRequiredSherlockFragmentActi
composeText = (EditText)findViewById(R.id.embedded_text_editor);
masterSecret = getIntent().getParcelableExtra(MASTER_SECRET_EXTRA);
charactersLeft = (TextView)findViewById(R.id.space_left);
+ emojiDrawer = (EmojiDrawer)findViewById(R.id.emoji_drawer);
+ emojiToggle = (EmojiToggle)findViewById(R.id.emoji_toggle);
attachmentAdapter = new AttachmentTypeSelectorAdapter(this);
attachmentManager = new AttachmentManager(this);
- SendButtonListener sendButtonListener = new SendButtonListener();
+ SendButtonListener sendButtonListener = new SendButtonListener();
+ ComposeKeyPressedListener composeKeyPressedListener = new ComposeKeyPressedListener();
recipientsPanel.setPanelChangeListener(new RecipientsPanelChangeListener());
sendButton.setOnClickListener(sendButtonListener);
sendButton.setEnabled(true);
addContactButton.setOnClickListener(new AddRecipientButtonListener());
- composeText.setOnKeyListener(new ComposeKeyPressedListener());
- composeText.addTextChangedListener(new OnTextChangedListener());
+ composeText.setOnKeyListener(composeKeyPressedListener);
+ composeText.addTextChangedListener(composeKeyPressedListener);
composeText.setOnEditorActionListener(sendButtonListener);
+ composeText.setOnClickListener(composeKeyPressedListener);
+ emojiDrawer.setComposeEditText(composeText);
+ emojiToggle.setOnClickListener(new EmojiToggleListener());
registerForContextMenu(sendButton);
@@ -729,7 +750,7 @@ public class ConversationActivity extends PassphraseRequiredSherlockFragmentActi
}
private void calculateCharactersRemaining() {
- int charactersSpent = composeText.getText().length();
+ int charactersSpent = composeText.getText().toString().length();
CharacterCalculator.CharacterState characterState = characterCalculator.calculateCharacters(charactersSpent);
charactersLeft.setText(characterState.charactersRemaining + "/" + characterState.maxMessageSize + " (" + characterState.messagesSpent + ")");
}
@@ -873,6 +894,21 @@ public class ConversationActivity extends PassphraseRequiredSherlockFragmentActi
}
}
+ private class EmojiToggleListener implements OnClickListener {
+ @Override
+ public void onClick(View v) {
+ InputMethodManager input = (InputMethodManager)getSystemService(Context.INPUT_METHOD_SERVICE);
+
+ if (emojiDrawer.getVisibility() == View.VISIBLE) {
+ input.showSoftInput(composeText, 0);
+ emojiDrawer.setVisibility(View.GONE);
+ } else {
+ input.hideSoftInputFromWindow(composeText.getWindowToken(), 0);
+ emojiDrawer.setVisibility(View.VISIBLE);
+ }
+ }
+ }
+
private class SendButtonListener implements OnClickListener, TextView.OnEditorActionListener {
@Override
public void onClick(View v) {
@@ -890,7 +926,29 @@ public class ConversationActivity extends PassphraseRequiredSherlockFragmentActi
}
}
- private class OnTextChangedListener implements TextWatcher {
+ private class ComposeKeyPressedListener implements OnKeyListener, OnClickListener, TextWatcher {
+ @Override
+ public boolean onKey(View v, int keyCode, KeyEvent event) {
+ if (event.getAction() == KeyEvent.ACTION_DOWN) {
+ if (keyCode == KeyEvent.KEYCODE_ENTER) {
+ if (PreferenceManager.getDefaultSharedPreferences(ConversationActivity.this).getBoolean("pref_enter_sends", false)) {
+ sendButton.dispatchKeyEvent(new KeyEvent(KeyEvent.ACTION_DOWN, KeyEvent.KEYCODE_ENTER));
+ sendButton.dispatchKeyEvent(new KeyEvent(KeyEvent.ACTION_UP, KeyEvent.KEYCODE_ENTER));
+ return true;
+ }
+ }
+ }
+
+ return false;
+ }
+
+ @Override
+ public void onClick(View v) {
+ if (emojiDrawer.isOpen()) {
+ emojiToggle.performClick();
+ }
+ }
+
@Override
public void afterTextChanged(Editable s) {
calculateCharactersRemaining();
@@ -908,28 +966,11 @@ public class ConversationActivity extends PassphraseRequiredSherlockFragmentActi
public void beforeTextChanged(CharSequence s, int start, int count,int after) {}
@Override
public void onTextChanged(CharSequence s, int start, int before,int count) {}
-
- }
-
- private class ComposeKeyPressedListener implements OnKeyListener {
- @Override
- public boolean onKey(View v, int keyCode, KeyEvent event) {
- if (event.getAction() == KeyEvent.ACTION_DOWN) {
- if (keyCode == KeyEvent.KEYCODE_ENTER) {
- if (PreferenceManager.getDefaultSharedPreferences(ConversationActivity.this).getBoolean("pref_enter_sends", false)) {
- sendButton.dispatchKeyEvent(new KeyEvent(KeyEvent.ACTION_DOWN, KeyEvent.KEYCODE_ENTER));
- sendButton.dispatchKeyEvent(new KeyEvent(KeyEvent.ACTION_UP, KeyEvent.KEYCODE_ENTER));
- return true;
- }
- }
- }
-
- return false;
- }
}
@Override
public void setComposeText(String text) {
this.composeText.setText(text);
}
+
}
diff --git a/src/org/thoughtcrime/securesms/components/EmojiDrawer.java b/src/org/thoughtcrime/securesms/components/EmojiDrawer.java
new file mode 100644
index 0000000000..7470ec4ba5
--- /dev/null
+++ b/src/org/thoughtcrime/securesms/components/EmojiDrawer.java
@@ -0,0 +1,184 @@
+package org.thoughtcrime.securesms.components;
+
+import android.content.Context;
+import android.graphics.Color;
+import android.graphics.drawable.Drawable;
+import android.support.v4.view.PagerAdapter;
+import android.support.v4.view.PagerTabStrip;
+import android.support.v4.view.ViewPager;
+import android.text.SpannableString;
+import android.text.Spanned;
+import android.text.style.ImageSpan;
+import android.util.AttributeSet;
+import android.view.LayoutInflater;
+import android.view.View;
+import android.view.ViewGroup;
+import android.widget.AdapterView;
+import android.widget.BaseAdapter;
+import android.widget.EditText;
+import android.widget.FrameLayout;
+import android.widget.GridView;
+import android.widget.ImageView;
+import android.widget.TextView;
+
+import org.thoughtcrime.securesms.R;
+import org.thoughtcrime.securesms.util.Emoji;
+
+import java.io.IOException;
+
+public class EmojiDrawer extends FrameLayout {
+
+ private FrameLayout emojiGridLayout;
+ private FrameLayout recentEmojiGridLayout;
+
+ private EditText composeText;
+
+ public EmojiDrawer(Context context) {
+ super(context);
+ initialize();
+ }
+
+ public EmojiDrawer(Context context, AttributeSet attrs) {
+ super(context, attrs);
+ initialize();
+ }
+
+ public EmojiDrawer(Context context, AttributeSet attrs, int defStyle) {
+ super(context, attrs, defStyle);
+ initialize();
+ }
+
+ public void setComposeEditText(EditText composeText) {
+ this.composeText = composeText;
+ }
+
+ public boolean isOpen() {
+ return getVisibility() == View.VISIBLE;
+ }
+
+ private void initialize() {
+ LayoutInflater inflater = (LayoutInflater)getContext().getSystemService(Context.LAYOUT_INFLATER_SERVICE);
+ inflater.inflate(R.layout.emoji_drawer, this, true);
+
+ ViewPager pager = (ViewPager ) findViewById(R.id.emoji_pager );
+ PagerTabStrip pagerTabStrip = (PagerTabStrip) findViewById(R.id.emoji_tab_strip);
+ this.emojiGridLayout = (FrameLayout) inflater.inflate(R.layout.emoji_grid_layout, null);
+ this.recentEmojiGridLayout = (FrameLayout) inflater.inflate(R.layout.emoji_grid_layout, null);
+ GridView emojiGrid = (GridView) emojiGridLayout.findViewById(R.id.emoji);
+ GridView recentEmojiGrid = (GridView) recentEmojiGridLayout.findViewById(R.id.emoji);
+
+ pagerTabStrip.setBackgroundColor(Color.parseColor("#ff333333"));
+ pagerTabStrip.setTextSpacing(1);
+ pager.setBackgroundColor(Color.parseColor("#ff333333"));
+ emojiGrid.setAdapter(new AllEmojiGridAdapter());
+ emojiGrid.setOnItemClickListener(new EmojiClickListener());
+ pager.setAdapter(new EmojiPagerAdapter());
+ pager.setCurrentItem(1);
+ }
+
+ private class EmojiClickListener implements AdapterView.OnItemClickListener {
+ @Override
+ public void onItemClick(AdapterView> parent, View view, int position, long id) {
+ int start = composeText.getSelectionStart();
+ int end = composeText.getSelectionEnd ();
+ String emoji = Emoji.EMOJI_ASSET_CODE_MAP.get(Emoji.EMOJI_ASSETS.get(position));
+
+ composeText.getText().replace(Math.min(start, end), Math.max(start, end),
+ emoji, 0, emoji.length());
+
+ composeText.setText(Emoji.emojify(getContext(), composeText.getText().toString()),
+ TextView.BufferType.SPANNABLE);
+
+ composeText.setSelection(end+2);
+ }
+ }
+
+ private class AllEmojiGridAdapter extends BaseAdapter {
+
+ @Override
+ public int getCount() {
+ return Emoji.EMOJI_ASSETS.size();
+ }
+
+ @Override
+ public Object getItem(int position) {
+ return null;
+ }
+
+ @Override
+ public long getItemId(int position) {
+ return 0;
+ }
+
+ @Override
+ public View getView(int position, View convertView, ViewGroup parent) {
+ try {
+ String asset = Emoji.EMOJI_ASSETS.get(position);
+ Drawable drawable = Drawable.createFromStream(getContext().getAssets().open(asset + ".png"), null);
+
+ if (convertView != null && convertView instanceof ImageView) {
+ ((ImageView)convertView).setImageDrawable(drawable);
+ return convertView;
+ } else {
+ ImageView imageView = new ImageView(getContext());
+ imageView.setImageDrawable(drawable);
+ return imageView;
+ }
+ } catch (IOException ioe) {
+ throw new AssertionError(ioe);
+ }
+ }
+ }
+
+
+ private class EmojiPagerAdapter extends PagerAdapter {
+
+ @Override
+ public int getCount() {
+ return 2;
+ }
+
+ @Override
+ public CharSequence getPageTitle(int position) {
+ switch (position) {
+ case 0:
+ SpannableString recent = new SpannableString(" Recent ");
+ ImageSpan recentImage = new ImageSpan(getContext(), R.drawable.ic_emoji_recent_light,
+ ImageSpan.ALIGN_BASELINE);
+
+ recent.setSpan(recentImage, 1, recent.length()-1, Spanned.SPAN_EXCLUSIVE_EXCLUSIVE);
+
+ return recent;
+ case 1:
+ SpannableString emoji = new SpannableString(" Emoji ");
+ ImageSpan emojiImage = new ImageSpan(getContext(), R.drawable.ic_emoji_light,
+ ImageSpan.ALIGN_BASELINE);
+
+ emoji.setSpan(emojiImage, 1, emoji.length()-1, Spanned.SPAN_EXCLUSIVE_EXCLUSIVE);
+
+ return emoji;
+ default:
+ throw new AssertionError("Bad position!");
+ }
+ }
+
+ @Override
+ public boolean isViewFromObject(View view, Object o) {
+ return view == o;
+ }
+
+ public Object instantiateItem(ViewGroup container, int position) {
+ View view;
+
+ switch (position) {
+ case 0: view = recentEmojiGridLayout; break;
+ case 1: view = emojiGridLayout; break;
+ default: throw new AssertionError("Too many positions!");
+ }
+
+ container.addView(view, 0);
+
+ return view;
+ }
+ }
+}
diff --git a/src/org/thoughtcrime/securesms/components/EmojiToggle.java b/src/org/thoughtcrime/securesms/components/EmojiToggle.java
new file mode 100644
index 0000000000..5d51b8f166
--- /dev/null
+++ b/src/org/thoughtcrime/securesms/components/EmojiToggle.java
@@ -0,0 +1,76 @@
+package org.thoughtcrime.securesms.components;
+
+
+import android.content.Context;
+import android.content.res.TypedArray;
+import android.graphics.drawable.Drawable;
+import android.util.AttributeSet;
+import android.view.View;
+import android.widget.ImageButton;
+
+import org.thoughtcrime.securesms.R;
+
+public class EmojiToggle extends ImageButton {
+
+ private Drawable emojiToggle;
+ private Drawable imeToggle;
+ private OnClickListener listener;
+
+ public EmojiToggle(Context context) {
+ super(context);
+ initialize();
+ }
+
+ public EmojiToggle(Context context, AttributeSet attrs) {
+ super(context, attrs);
+ initialize();
+ }
+
+ public EmojiToggle(Context context, AttributeSet attrs, int defStyle) {
+ super(context, attrs, defStyle);
+ initialize();
+ }
+
+ @Override
+ public void setOnClickListener(OnClickListener listener) {
+ this.listener = listener;
+ }
+
+ public void toggle() {
+ if (getDrawable() == emojiToggle) {
+ setImageDrawable(imeToggle);
+ } else {
+ setImageDrawable(emojiToggle);
+ }
+ }
+
+ private void initialize() {
+ initializeResources();
+ initializeListeners();
+ }
+
+ private void initializeResources() {
+ int attributes[] = new int[] {R.attr.conversation_emoji_toggle,
+ R.attr.conversation_keyboard_toggle};
+
+ TypedArray drawables = getContext().obtainStyledAttributes(attributes);
+ this.emojiToggle = drawables.getDrawable(0);
+ this.imeToggle = drawables.getDrawable(1);
+
+ drawables.recycle();
+
+ setImageDrawable(this.emojiToggle);
+ }
+
+ private void initializeListeners() {
+ super.setOnClickListener(new OnClickListener() {
+ @Override
+ public void onClick(View v) {
+ toggle();
+
+ if (listener != null)
+ listener.onClick(v);
+ }
+ });
+ }
+}
diff --git a/src/org/thoughtcrime/securesms/util/Emoji.java b/src/org/thoughtcrime/securesms/util/Emoji.java
new file mode 100644
index 0000000000..1472f44b42
--- /dev/null
+++ b/src/org/thoughtcrime/securesms/util/Emoji.java
@@ -0,0 +1,176 @@
+package org.thoughtcrime.securesms.util;
+
+import android.content.Context;
+import android.graphics.drawable.Drawable;
+import android.text.Spannable;
+import android.text.SpannableString;
+import android.text.style.ImageSpan;
+import android.util.Log;
+
+import java.io.IOException;
+import java.util.ArrayList;
+import java.util.HashMap;
+import java.util.Map;
+import java.util.regex.Matcher;
+import java.util.regex.Pattern;
+
+public class Emoji {
+
+ public static final HashMap EMOJI_ASSET_CODE_MAP = new HashMap();
+ public static final HashMap EMOJI_CODE_ASSET_MAP = new HashMap();
+
+ public static final ArrayList EMOJI_ASSETS = new ArrayList() {{
+ add("1f601");
+ add("1f602");
+ add("1f603");
+ add("1f604");
+ add("1f605");
+ add("1f606");
+ add("1f609");
+ add("1f60a");
+ add("1f60b");
+ add("1f60c");
+ add("1f60d");
+ add("1f60f");
+ add("1f612");
+ add("1f613");
+ add("1f614");
+ add("1f616");
+ add("1f618");
+ add("1f61a");
+ add("1f61c");
+ add("1f61d");
+ add("1f61e");
+ add("1f620");
+ add("1f621");
+ add("1f622");
+ add("1f623");
+ add("1f624");
+ add("1f625");
+ add("1f628");
+ add("1f629");
+ add("1f62a");
+ add("1f62b");
+ add("1f62d");
+ add("1f630");
+ add("1f631");
+ add("1f632");
+ add("1f633");
+ add("1f635");
+ add("1f637");
+
+ add("1f638");
+ add("1f639");
+ add("1f63a");
+ add("1f63b");
+ add("1f63c");
+ add("1f63d");
+ add("1f63e");
+ add("1f63f");
+ add("1f640");
+ add("1f645");
+ add("1f646");
+ add("1f647");
+ add("1f648");
+ add("1f649");
+ add("1f64a");
+ add("1f64b");
+ add("1f64c");
+ add("1f64d");
+ add("1f64e");
+ add("1f64f");
+ }};
+
+ public static ArrayList EMOJI_CODES = new ArrayList() {{
+ add("\ud83d\ude01");
+ add("\ud83d\ude02");
+ add("\ud83d\ude03");
+ add("\ud83d\ude04");
+ add("\ud83d\ude05");
+ add("\ud83d\ude06");
+ add("\ud83d\ude09");
+ add("\ud83d\ude0a");
+ add("\ud83d\ude0b");
+ add("\ud83d\ude0c");
+ add("\ud83d\ude0d");
+ add("\ud83d\ude0f");
+ add("\ud83d\ude12");
+ add("\ud83d\ude13");
+ add("\ud83d\ude14");
+ add("\ud83d\ude16");
+ add("\ud83d\ude18");
+ add("\ud83d\ude1a");
+ add("\ud83d\ude1c");
+ add("\ud83d\ude1d");
+ add("\ud83d\ude1e");
+ add("\ud83d\ude20");
+ add("\ud83d\ude21");
+ add("\ud83d\ude22");
+ add("\ud83d\ude23");
+ add("\ud83d\ude24");
+ add("\ud83d\ude25");
+ add("\ud83d\ude28");
+ add("\ud83d\ude29");
+ add("\ud83d\ude2a");
+ add("\ud83d\ude2b");
+ add("\ud83d\ude2d");
+ add("\ud83d\ude30");
+ add("\ud83d\ude31");
+ add("\ud83d\ude32");
+ add("\ud83d\ude33");
+ add("\ud83d\ude35");
+ add("\ud83d\ude37");
+
+ add("\ud83d\ude38");
+ add("\ud83d\ude39");
+ add("\ud83d\ude3a");
+ add("\ud83d\ude3b");
+ add("\ud83d\ude3c");
+ add("\ud83d\ude3d");
+ add("\ud83d\ude3e");
+ add("\ud83d\ude3f");
+ add("\ud83d\ude40");
+ add("\ud83d\ude45");
+ add("\ud83d\ude46");
+ add("\ud83d\ude47");
+ add("\ud83d\ude48");
+ add("\ud83d\ude49");
+ add("\ud83d\ude4a");
+ add("\ud83d\ude4b");
+ add("\ud83d\ude4c");
+ add("\ud83d\ude4d");
+ add("\ud83d\ude4e");
+ add("\ud83d\ude4f");
+ }};
+
+ static {
+ for (int i=0;i entry : EMOJI_CODE_ASSET_MAP.entrySet()) {
+ Matcher matcher = entry.getKey().matcher(spannable);
+
+ while (matcher.find()) {
+ Drawable asset = Drawable.createFromStream(context.getAssets().open(entry.getValue() + ".png"), null);
+ asset.setBounds(0, 0, asset.getIntrinsicWidth(), asset.getIntrinsicHeight());
+
+ ImageSpan imageSpan = new ImageSpan(asset, ImageSpan.ALIGN_BASELINE);
+ Log.w("Emoji", "Replacing text with: " + imageSpan);
+ spannable.setSpan(imageSpan,matcher.start(), matcher.end(),
+ Spannable.SPAN_EXCLUSIVE_EXCLUSIVE);
+ }
+ }
+
+ return spannable;
+ } catch (IOException e) {
+ throw new AssertionError(e);
+ }
+ }
+}