mirror of
https://github.com/oxen-io/session-android.git
synced 2025-01-11 20:53:39 +00:00
Voice Note Locking.
Limit of 60 minutes, after which it's cancelled.
This commit is contained in:
parent
cab3657ab0
commit
e842f78457
9
res/drawable/ic_chevron_up.xml
Normal file
9
res/drawable/ic_chevron_up.xml
Normal file
@ -0,0 +1,9 @@
|
|||||||
|
<vector xmlns:android="http://schemas.android.com/apk/res/android"
|
||||||
|
android:width="24dp"
|
||||||
|
android:height="24dp"
|
||||||
|
android:viewportWidth="24"
|
||||||
|
android:viewportHeight="24">
|
||||||
|
<path
|
||||||
|
android:fillColor="#FF000000"
|
||||||
|
android:pathData="M22.1,15l-9.5,-9.5c-0.3,-0.3 -0.8,-0.3 -1.1,0L1.9,15L3,16.1l9,-9l9,9L22.1,15z"/>
|
||||||
|
</vector>
|
9
res/drawable/ic_lock.xml
Normal file
9
res/drawable/ic_lock.xml
Normal file
@ -0,0 +1,9 @@
|
|||||||
|
<vector xmlns:android="http://schemas.android.com/apk/res/android"
|
||||||
|
android:width="24dp"
|
||||||
|
android:height="24dp"
|
||||||
|
android:viewportWidth="24"
|
||||||
|
android:viewportHeight="24">
|
||||||
|
<path
|
||||||
|
android:fillColor="#FF000000"
|
||||||
|
android:pathData="M18,9h-1V6c0,-2.8 -2.2,-5 -5,-5S7,3.2 7,6v3H6c-1.1,0 -2,0.9 -2,2v9c0,1.1 0.9,2 2,2h12c1.1,0 2,-0.9 2,-2v-9C20,9.9 19.1,9 18,9zM9,6c0,-1.7 1.3,-3 3,-3s3,1.3 3,3v3H9V6zM13,16.2v2.3h-2v-2.3c-0.6,-0.4 -1,-1 -1,-1.7c0,-1.1 0.9,-2 2,-2s2,0.9 2,2C14,15.2 13.6,15.9 13,16.2z"/>
|
||||||
|
</vector>
|
@ -124,31 +124,12 @@
|
|||||||
<org.thoughtcrime.securesms.components.MicrophoneRecorderView
|
<org.thoughtcrime.securesms.components.MicrophoneRecorderView
|
||||||
android:id="@+id/recorder_view"
|
android:id="@+id/recorder_view"
|
||||||
android:layout_height="match_parent"
|
android:layout_height="match_parent"
|
||||||
android:layout_width="wrap_content"
|
android:layout_width="36dp"
|
||||||
android:layout_gravity="center_vertical"
|
android:layout_gravity="center_vertical"
|
||||||
android:clipChildren="false"
|
android:clipChildren="false"
|
||||||
android:clipToPadding="false">
|
android:clipToPadding="false">
|
||||||
|
|
||||||
<ImageButton
|
<include layout="@layout/microphone_recorder_view" />
|
||||||
android:id="@+id/quick_audio_toggle"
|
|
||||||
android:layout_width="wrap_content"
|
|
||||||
android:layout_height="match_parent"
|
|
||||||
android:layout_gravity="center_vertical"
|
|
||||||
android:layout_marginEnd="2dp"
|
|
||||||
android:padding="6dp"
|
|
||||||
android:src="?quick_mic_icon"
|
|
||||||
android:background="@null"
|
|
||||||
android:contentDescription="@string/conversation_activity__quick_attachment_drawer_record_and_send_audio_description" />
|
|
||||||
|
|
||||||
<ImageView
|
|
||||||
android:id="@+id/quick_audio_fab"
|
|
||||||
android:layout_width="74dp"
|
|
||||||
android:layout_height="74dp"
|
|
||||||
android:src="@drawable/ic_mic_white_48dp"
|
|
||||||
android:background="@drawable/circle_tintable"
|
|
||||||
android:backgroundTint="@color/core_red"
|
|
||||||
android:visibility="gone"
|
|
||||||
android:scaleType="center"/>
|
|
||||||
|
|
||||||
</org.thoughtcrime.securesms.components.MicrophoneRecorderView>
|
</org.thoughtcrime.securesms.components.MicrophoneRecorderView>
|
||||||
|
|
||||||
@ -177,51 +158,7 @@
|
|||||||
|
|
||||||
</LinearLayout>
|
</LinearLayout>
|
||||||
|
|
||||||
<LinearLayout
|
<include layout="@layout/recording_layout" />
|
||||||
android:id="@+id/recording_container"
|
|
||||||
android:layout_width="match_parent"
|
|
||||||
android:layout_height="wrap_content"
|
|
||||||
android:layout_gravity="center_vertical"
|
|
||||||
android:gravity="center_vertical"
|
|
||||||
android:orientation="horizontal"
|
|
||||||
tools:visibility="gone">
|
|
||||||
|
|
||||||
<TextView
|
|
||||||
android:id="@+id/record_time"
|
|
||||||
android:layout_width="wrap_content"
|
|
||||||
android:layout_height="wrap_content"
|
|
||||||
android:ellipsize="none"
|
|
||||||
android:layout_marginStart="20dp"
|
|
||||||
style="@style/Signal.Text.Body"
|
|
||||||
android:text="00:00"
|
|
||||||
android:textColor="@color/core_grey_60"
|
|
||||||
android:singleLine="true"
|
|
||||||
android:visibility="gone"
|
|
||||||
tools:visibility="visible"/>
|
|
||||||
|
|
||||||
<FrameLayout
|
|
||||||
android:layout_width="match_parent"
|
|
||||||
android:layout_height="wrap_content"
|
|
||||||
android:clipChildren="true">
|
|
||||||
|
|
||||||
<TextView
|
|
||||||
android:id="@+id/slide_to_cancel"
|
|
||||||
android:layout_width="wrap_content"
|
|
||||||
android:layout_height="wrap_content"
|
|
||||||
android:gravity="left|start|center_vertical"
|
|
||||||
android:drawableStart="@drawable/ic_keyboard_arrow_left_grey600_24dp"
|
|
||||||
style="@style/Signal.Text.Caption"
|
|
||||||
android:text="@string/conversation_input_panel__slide_to_cancel"
|
|
||||||
android:textAllCaps="true"
|
|
||||||
android:textColor="@color/core_grey_60"
|
|
||||||
android:ellipsize="none"
|
|
||||||
android:visibility="gone"
|
|
||||||
android:paddingEnd="50dp"
|
|
||||||
tools:visibility="visible"/>
|
|
||||||
|
|
||||||
</FrameLayout>
|
|
||||||
|
|
||||||
</LinearLayout>
|
|
||||||
|
|
||||||
</FrameLayout>
|
</FrameLayout>
|
||||||
|
|
||||||
|
63
res/layout/microphone_recorder_view.xml
Normal file
63
res/layout/microphone_recorder_view.xml
Normal file
@ -0,0 +1,63 @@
|
|||||||
|
<?xml version="1.0" encoding="utf-8"?>
|
||||||
|
<merge xmlns:android="http://schemas.android.com/apk/res/android"
|
||||||
|
xmlns:app="http://schemas.android.com/apk/res-auto"
|
||||||
|
xmlns:tools="http://schemas.android.com/tools"
|
||||||
|
tools:showIn="@layout/conversation_input_panel">
|
||||||
|
|
||||||
|
<ImageButton
|
||||||
|
android:id="@+id/quick_audio_toggle"
|
||||||
|
android:layout_width="wrap_content"
|
||||||
|
android:layout_height="match_parent"
|
||||||
|
android:layout_gravity="center_vertical"
|
||||||
|
android:layout_marginEnd="2dp"
|
||||||
|
android:background="@null"
|
||||||
|
android:contentDescription="@string/conversation_activity__quick_attachment_drawer_record_and_send_audio_description"
|
||||||
|
android:padding="6dp"
|
||||||
|
android:src="?quick_mic_icon" />
|
||||||
|
|
||||||
|
<LinearLayout
|
||||||
|
android:id="@+id/lock_drop_target"
|
||||||
|
android:layout_width="40dp"
|
||||||
|
android:layout_height="64dp"
|
||||||
|
android:layout_gravity="center"
|
||||||
|
android:background="?attr/conversation_input_background"
|
||||||
|
android:clipChildren="false"
|
||||||
|
android:orientation="vertical"
|
||||||
|
android:visibility="gone"
|
||||||
|
tools:showIn="@layout/conversation_input_panel"
|
||||||
|
tools:translationY="@dimen/recording_voice_lock_target"
|
||||||
|
tools:visibility="visible">
|
||||||
|
|
||||||
|
<ImageView
|
||||||
|
android:layout_width="24dp"
|
||||||
|
android:layout_height="24dp"
|
||||||
|
android:layout_gravity="center_horizontal"
|
||||||
|
android:layout_marginTop="10dp"
|
||||||
|
android:contentDescription="@string/conversation_activity__quick_attachment_drawer_lock_record_description"
|
||||||
|
android:tint="?attr/conversation_input_inline_attach_icon_tint"
|
||||||
|
app:srcCompat="@drawable/ic_lock" />
|
||||||
|
|
||||||
|
<ImageView
|
||||||
|
android:layout_width="18dp"
|
||||||
|
android:layout_height="18dp"
|
||||||
|
android:layout_gravity="center_horizontal"
|
||||||
|
android:layout_marginTop="6dp"
|
||||||
|
android:contentDescription="@string/conversation_activity__quick_attachment_drawer_lock_record_description"
|
||||||
|
android:tint="?attr/conversation_input_inline_attach_icon_tint"
|
||||||
|
app:srcCompat="@drawable/ic_chevron_up" />
|
||||||
|
</LinearLayout>
|
||||||
|
|
||||||
|
<ImageView
|
||||||
|
android:id="@+id/quick_audio_fab"
|
||||||
|
android:layout_width="74dp"
|
||||||
|
android:layout_height="74dp"
|
||||||
|
android:layout_marginStart="-18dp"
|
||||||
|
android:layout_marginTop="-18dp"
|
||||||
|
android:background="@drawable/circle_tintable"
|
||||||
|
android:clickable="false"
|
||||||
|
android:scaleType="center"
|
||||||
|
android:src="@drawable/ic_mic_white_48dp"
|
||||||
|
android:visibility="gone"
|
||||||
|
tools:ignore="ContentDescription"
|
||||||
|
tools:visibility="visible" />
|
||||||
|
</merge>
|
74
res/layout/recording_layout.xml
Normal file
74
res/layout/recording_layout.xml
Normal file
@ -0,0 +1,74 @@
|
|||||||
|
<?xml version="1.0" encoding="utf-8"?>
|
||||||
|
<android.support.constraint.ConstraintLayout xmlns:android="http://schemas.android.com/apk/res/android"
|
||||||
|
xmlns:app="http://schemas.android.com/apk/res-auto"
|
||||||
|
xmlns:tools="http://schemas.android.com/tools"
|
||||||
|
android:id="@+id/recording_container"
|
||||||
|
android:layout_width="match_parent"
|
||||||
|
android:layout_height="wrap_content"
|
||||||
|
android:layout_gravity="center_vertical"
|
||||||
|
android:gravity="center_vertical"
|
||||||
|
android:orientation="horizontal"
|
||||||
|
tools:visibility="visible"
|
||||||
|
tools:showIn="@layout/conversation_input_panel">
|
||||||
|
|
||||||
|
<TextView
|
||||||
|
android:id="@+id/record_time"
|
||||||
|
style="@style/Signal.Text.Body"
|
||||||
|
android:layout_width="wrap_content"
|
||||||
|
android:layout_height="wrap_content"
|
||||||
|
android:layout_marginStart="16dp"
|
||||||
|
android:ellipsize="none"
|
||||||
|
android:singleLine="true"
|
||||||
|
android:textColor="@color/core_grey_60"
|
||||||
|
android:visibility="gone"
|
||||||
|
app:layout_constraintBottom_toBottomOf="parent"
|
||||||
|
app:layout_constraintStart_toStartOf="parent"
|
||||||
|
app:layout_constraintTop_toTopOf="parent"
|
||||||
|
tools:text="00:00"
|
||||||
|
tools:visibility="visible" />
|
||||||
|
|
||||||
|
<TextView
|
||||||
|
android:id="@+id/record_cancel"
|
||||||
|
style="@style/Signal.Text.Caption"
|
||||||
|
android:layout_width="wrap_content"
|
||||||
|
android:layout_height="wrap_content"
|
||||||
|
android:layout_marginStart="8dp"
|
||||||
|
android:layout_marginEnd="8dp"
|
||||||
|
android:ellipsize="none"
|
||||||
|
android:gravity="center_vertical"
|
||||||
|
android:text="@string/conversation_input_panel__cancel"
|
||||||
|
android:textAllCaps="true"
|
||||||
|
android:textColor="@color/red_A700"
|
||||||
|
android:visibility="gone"
|
||||||
|
app:layout_constraintBottom_toBottomOf="parent"
|
||||||
|
app:layout_constraintEnd_toEndOf="parent"
|
||||||
|
app:layout_constraintStart_toStartOf="@+id/frameLayout"
|
||||||
|
app:layout_constraintTop_toTopOf="parent"
|
||||||
|
tools:visibility="visible" />
|
||||||
|
|
||||||
|
<FrameLayout
|
||||||
|
android:id="@+id/frameLayout"
|
||||||
|
android:layout_width="wrap_content"
|
||||||
|
android:layout_height="match_parent"
|
||||||
|
android:clipChildren="true"
|
||||||
|
app:layout_constraintBottom_toBottomOf="parent"
|
||||||
|
app:layout_constraintStart_toEndOf="@+id/record_time"
|
||||||
|
app:layout_constraintTop_toTopOf="parent">
|
||||||
|
|
||||||
|
<TextView
|
||||||
|
android:id="@+id/slide_to_cancel"
|
||||||
|
style="@style/Signal.Text.Caption"
|
||||||
|
android:layout_width="match_parent"
|
||||||
|
android:layout_height="wrap_content"
|
||||||
|
android:drawableStart="@drawable/ic_keyboard_arrow_left_grey600_24dp"
|
||||||
|
android:ellipsize="none"
|
||||||
|
android:gravity="center_vertical"
|
||||||
|
android:text="@string/conversation_input_panel__slide_to_cancel"
|
||||||
|
android:textAllCaps="true"
|
||||||
|
android:textColor="@color/core_grey_60"
|
||||||
|
android:visibility="gone"
|
||||||
|
tools:visibility="visible" />
|
||||||
|
|
||||||
|
</FrameLayout>
|
||||||
|
|
||||||
|
</android.support.constraint.ConstraintLayout>
|
@ -99,4 +99,6 @@
|
|||||||
|
|
||||||
<dimen name="alertview_small_icon_size">14dp</dimen>
|
<dimen name="alertview_small_icon_size">14dp</dimen>
|
||||||
|
|
||||||
|
<dimen name="recording_voice_lock_target">-150dp</dimen>
|
||||||
|
|
||||||
</resources>
|
</resources>
|
||||||
|
@ -869,10 +869,12 @@
|
|||||||
<string name="conversation_activity__attachment_thumbnail">Attachment Thumbnail</string>
|
<string name="conversation_activity__attachment_thumbnail">Attachment Thumbnail</string>
|
||||||
<string name="conversation_activity__quick_attachment_drawer_toggle_camera_description">Toggle quick camera attachment drawer</string>
|
<string name="conversation_activity__quick_attachment_drawer_toggle_camera_description">Toggle quick camera attachment drawer</string>
|
||||||
<string name="conversation_activity__quick_attachment_drawer_record_and_send_audio_description">Record and send audio attachment</string>
|
<string name="conversation_activity__quick_attachment_drawer_record_and_send_audio_description">Record and send audio attachment</string>
|
||||||
|
<string name="conversation_activity__quick_attachment_drawer_lock_record_description">Lock recording of audio attachment</string>
|
||||||
<string name="conversation_activity__enable_signal_for_sms">Enable Signal for SMS</string>
|
<string name="conversation_activity__enable_signal_for_sms">Enable Signal for SMS</string>
|
||||||
|
|
||||||
<!-- conversation_input_panel -->
|
<!-- conversation_input_panel -->
|
||||||
<string name="conversation_input_panel__slide_to_cancel">SLIDE TO CANCEL</string>
|
<string name="conversation_input_panel__slide_to_cancel">Slide to cancel</string>
|
||||||
|
<string name="conversation_input_panel__cancel">Cancel</string>
|
||||||
|
|
||||||
<!-- conversation_item -->
|
<!-- conversation_item -->
|
||||||
<string name="conversation_item__mms_image_description">Media message</string>
|
<string name="conversation_item__mms_image_description">Media message</string>
|
||||||
|
@ -5,6 +5,7 @@ import android.content.Context;
|
|||||||
import android.net.Uri;
|
import android.net.Uri;
|
||||||
import android.os.Build;
|
import android.os.Build;
|
||||||
import android.support.annotation.DimenRes;
|
import android.support.annotation.DimenRes;
|
||||||
|
import android.support.annotation.MainThread;
|
||||||
import android.support.annotation.NonNull;
|
import android.support.annotation.NonNull;
|
||||||
import android.support.annotation.Nullable;
|
import android.support.annotation.Nullable;
|
||||||
import android.support.v4.view.ViewCompat;
|
import android.support.v4.view.ViewCompat;
|
||||||
@ -38,7 +39,6 @@ import org.thoughtcrime.securesms.util.concurrent.SettableFuture;
|
|||||||
import org.whispersystems.libsignal.util.guava.Optional;
|
import org.whispersystems.libsignal.util.guava.Optional;
|
||||||
|
|
||||||
import java.util.concurrent.TimeUnit;
|
import java.util.concurrent.TimeUnit;
|
||||||
import java.util.concurrent.atomic.AtomicLong;
|
|
||||||
|
|
||||||
public class InputPanel extends LinearLayout
|
public class InputPanel extends LinearLayout
|
||||||
implements MicrophoneRecorderView.Listener,
|
implements MicrophoneRecorderView.Listener,
|
||||||
@ -58,6 +58,7 @@ public class InputPanel extends LinearLayout
|
|||||||
private View quickAudioToggle;
|
private View quickAudioToggle;
|
||||||
private View buttonToggle;
|
private View buttonToggle;
|
||||||
private View recordingContainer;
|
private View recordingContainer;
|
||||||
|
private View recordLockCancel;
|
||||||
|
|
||||||
private MicrophoneRecorderView microphoneRecorderView;
|
private MicrophoneRecorderView microphoneRecorderView;
|
||||||
private SlideToCancel slideToCancel;
|
private SlideToCancel slideToCancel;
|
||||||
@ -93,10 +94,15 @@ public class InputPanel extends LinearLayout
|
|||||||
this.quickAudioToggle = findViewById(R.id.quick_audio_toggle);
|
this.quickAudioToggle = findViewById(R.id.quick_audio_toggle);
|
||||||
this.buttonToggle = findViewById(R.id.button_toggle);
|
this.buttonToggle = findViewById(R.id.button_toggle);
|
||||||
this.recordingContainer = findViewById(R.id.recording_container);
|
this.recordingContainer = findViewById(R.id.recording_container);
|
||||||
this.recordTime = new RecordTime(findViewById(R.id.record_time));
|
this.recordLockCancel = findViewById(R.id.record_cancel);
|
||||||
this.slideToCancel = new SlideToCancel(findViewById(R.id.slide_to_cancel));
|
this.slideToCancel = new SlideToCancel(findViewById(R.id.slide_to_cancel));
|
||||||
this.microphoneRecorderView = findViewById(R.id.recorder_view);
|
this.microphoneRecorderView = findViewById(R.id.recorder_view);
|
||||||
this.microphoneRecorderView.setListener(this);
|
this.microphoneRecorderView.setListener(this);
|
||||||
|
this.recordTime = new RecordTime(findViewById(R.id.record_time),
|
||||||
|
TimeUnit.HOURS.toSeconds(1),
|
||||||
|
() -> microphoneRecorderView.cancelAction());
|
||||||
|
|
||||||
|
this.recordLockCancel.setOnClickListener(v -> microphoneRecorderView.cancelAction());
|
||||||
|
|
||||||
if (TextSecurePreferences.isSystemEmojiPreferred(getContext())) {
|
if (TextSecurePreferences.isSystemEmojiPreferred(getContext())) {
|
||||||
emojiToggle.setVisibility(View.GONE);
|
emojiToggle.setVisibility(View.GONE);
|
||||||
@ -181,21 +187,21 @@ public class InputPanel extends LinearLayout
|
|||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
public void onRecordPressed(float startPositionX) {
|
public void onRecordPressed() {
|
||||||
if (listener != null) listener.onRecorderStarted();
|
if (listener != null) listener.onRecorderStarted();
|
||||||
recordTime.display();
|
recordTime.display();
|
||||||
slideToCancel.display(startPositionX);
|
slideToCancel.display();
|
||||||
|
|
||||||
if (emojiVisible) ViewUtil.fadeOut(emojiToggle, FADE_TIME, View.INVISIBLE);
|
if (emojiVisible) ViewUtil.fadeOut(emojiToggle, FADE_TIME, View.INVISIBLE);
|
||||||
ViewUtil.fadeOut(composeText, FADE_TIME, View.INVISIBLE);
|
ViewUtil.fadeOut(composeText, FADE_TIME, View.INVISIBLE);
|
||||||
ViewUtil.fadeOut(quickCameraToggle, FADE_TIME, View.INVISIBLE);
|
ViewUtil.fadeOut(quickCameraToggle, FADE_TIME, View.INVISIBLE);
|
||||||
ViewUtil.fadeOut(quickAudioToggle, FADE_TIME, View.INVISIBLE);
|
ViewUtil.fadeOut(quickAudioToggle, FADE_TIME, View.INVISIBLE);
|
||||||
ViewUtil.fadeOut(buttonToggle, FADE_TIME, View.INVISIBLE);
|
buttonToggle.animate().alpha(0).setDuration(FADE_TIME).start();
|
||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
public void onRecordReleased(float x) {
|
public void onRecordReleased() {
|
||||||
long elapsedTime = onRecordHideEvent(x);
|
long elapsedTime = onRecordHideEvent();
|
||||||
|
|
||||||
if (listener != null) {
|
if (listener != null) {
|
||||||
Log.d(TAG, "Elapsed time: " + elapsedTime);
|
Log.d(TAG, "Elapsed time: " + elapsedTime);
|
||||||
@ -209,8 +215,8 @@ public class InputPanel extends LinearLayout
|
|||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
public void onRecordMoved(float x, float absoluteX) {
|
public void onRecordMoved(float offsetX, float absoluteX) {
|
||||||
slideToCancel.moveTo(x);
|
slideToCancel.moveTo(offsetX);
|
||||||
|
|
||||||
int direction = ViewCompat.getLayoutDirection(this);
|
int direction = ViewCompat.getLayoutDirection(this);
|
||||||
float position = absoluteX / recordingContainer.getWidth();
|
float position = absoluteX / recordingContainer.getWidth();
|
||||||
@ -223,11 +229,19 @@ public class InputPanel extends LinearLayout
|
|||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
public void onRecordCanceled(float x) {
|
public void onRecordCanceled() {
|
||||||
onRecordHideEvent(x);
|
onRecordHideEvent();
|
||||||
if (listener != null) listener.onRecorderCanceled();
|
if (listener != null) listener.onRecorderCanceled();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public void onRecordLocked() {
|
||||||
|
slideToCancel.hide();
|
||||||
|
recordLockCancel.setVisibility(View.VISIBLE);
|
||||||
|
buttonToggle.animate().alpha(1).setDuration(FADE_TIME).start();
|
||||||
|
if (listener != null) listener.onRecorderLocked();
|
||||||
|
}
|
||||||
|
|
||||||
public void onPause() {
|
public void onPause() {
|
||||||
this.microphoneRecorderView.cancelAction();
|
this.microphoneRecorderView.cancelAction();
|
||||||
}
|
}
|
||||||
@ -239,8 +253,10 @@ public class InputPanel extends LinearLayout
|
|||||||
quickCameraToggle.setEnabled(enabled);
|
quickCameraToggle.setEnabled(enabled);
|
||||||
}
|
}
|
||||||
|
|
||||||
private long onRecordHideEvent(float x) {
|
private long onRecordHideEvent() {
|
||||||
ListenableFuture<Void> future = slideToCancel.hide(x);
|
recordLockCancel.setVisibility(View.GONE);
|
||||||
|
|
||||||
|
ListenableFuture<Void> future = slideToCancel.hide();
|
||||||
long elapsedTime = recordTime.hide();
|
long elapsedTime = recordTime.hide();
|
||||||
|
|
||||||
future.addListener(new AssertedSuccessListener<Void>() {
|
future.addListener(new AssertedSuccessListener<Void>() {
|
||||||
@ -250,7 +266,7 @@ public class InputPanel extends LinearLayout
|
|||||||
ViewUtil.fadeIn(composeText, FADE_TIME);
|
ViewUtil.fadeIn(composeText, FADE_TIME);
|
||||||
ViewUtil.fadeIn(quickCameraToggle, FADE_TIME);
|
ViewUtil.fadeIn(quickCameraToggle, FADE_TIME);
|
||||||
ViewUtil.fadeIn(quickAudioToggle, FADE_TIME);
|
ViewUtil.fadeIn(quickAudioToggle, FADE_TIME);
|
||||||
ViewUtil.fadeIn(buttonToggle, FADE_TIME);
|
buttonToggle.animate().alpha(1).setDuration(FADE_TIME).start();
|
||||||
}
|
}
|
||||||
});
|
});
|
||||||
|
|
||||||
@ -276,9 +292,17 @@ public class InputPanel extends LinearLayout
|
|||||||
return getResources().getDimensionPixelSize(dimenRes);
|
return getResources().getDimensionPixelSize(dimenRes);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
public boolean isRecordingInLockedMode() {
|
||||||
|
return microphoneRecorderView.isRecordingLocked();
|
||||||
|
}
|
||||||
|
|
||||||
|
public void releaseRecordingLock() {
|
||||||
|
microphoneRecorderView.unlockAction();
|
||||||
|
}
|
||||||
|
|
||||||
public interface Listener {
|
public interface Listener {
|
||||||
void onRecorderStarted();
|
void onRecorderStarted();
|
||||||
|
void onRecorderLocked();
|
||||||
void onRecorderFinished();
|
void onRecorderFinished();
|
||||||
void onRecorderCanceled();
|
void onRecorderCanceled();
|
||||||
void onRecorderPermissionRequired();
|
void onRecorderPermissionRequired();
|
||||||
@ -290,23 +314,19 @@ public class InputPanel extends LinearLayout
|
|||||||
|
|
||||||
private final View slideToCancelView;
|
private final View slideToCancelView;
|
||||||
|
|
||||||
private float startPositionX;
|
SlideToCancel(View slideToCancelView) {
|
||||||
|
|
||||||
public SlideToCancel(View slideToCancelView) {
|
|
||||||
this.slideToCancelView = slideToCancelView;
|
this.slideToCancelView = slideToCancelView;
|
||||||
}
|
}
|
||||||
|
|
||||||
public void display(float startPositionX) {
|
public void display() {
|
||||||
this.startPositionX = startPositionX;
|
|
||||||
ViewUtil.fadeIn(this.slideToCancelView, FADE_TIME);
|
ViewUtil.fadeIn(this.slideToCancelView, FADE_TIME);
|
||||||
}
|
}
|
||||||
|
|
||||||
public ListenableFuture<Void> hide(float x) {
|
public ListenableFuture<Void> hide() {
|
||||||
final SettableFuture<Void> future = new SettableFuture<>();
|
final SettableFuture<Void> future = new SettableFuture<>();
|
||||||
float offset = getOffset(x);
|
|
||||||
|
|
||||||
AnimationSet animation = new AnimationSet(true);
|
AnimationSet animation = new AnimationSet(true);
|
||||||
animation.addAnimation(new TranslateAnimation(Animation.ABSOLUTE, offset,
|
animation.addAnimation(new TranslateAnimation(Animation.ABSOLUTE, slideToCancelView.getTranslationX(),
|
||||||
Animation.ABSOLUTE, 0,
|
Animation.ABSOLUTE, 0,
|
||||||
Animation.RELATIVE_TO_SELF, 0,
|
Animation.RELATIVE_TO_SELF, 0,
|
||||||
Animation.RELATIVE_TO_SELF, 0));
|
Animation.RELATIVE_TO_SELF, 0));
|
||||||
@ -323,8 +343,7 @@ public class InputPanel extends LinearLayout
|
|||||||
return future;
|
return future;
|
||||||
}
|
}
|
||||||
|
|
||||||
public void moveTo(float x) {
|
void moveTo(float offset) {
|
||||||
float offset = getOffset(x);
|
|
||||||
Animation animation = new TranslateAnimation(Animation.ABSOLUTE, offset,
|
Animation animation = new TranslateAnimation(Animation.ABSOLUTE, offset,
|
||||||
Animation.ABSOLUTE, offset,
|
Animation.ABSOLUTE, offset,
|
||||||
Animation.RELATIVE_TO_SELF, 0,
|
Animation.RELATIVE_TO_SELF, 0,
|
||||||
@ -336,49 +355,55 @@ public class InputPanel extends LinearLayout
|
|||||||
|
|
||||||
slideToCancelView.startAnimation(animation);
|
slideToCancelView.startAnimation(animation);
|
||||||
}
|
}
|
||||||
|
|
||||||
private float getOffset(float x) {
|
|
||||||
return ViewCompat.getLayoutDirection(slideToCancelView) == ViewCompat.LAYOUT_DIRECTION_LTR ?
|
|
||||||
-Math.max(0, this.startPositionX - x) : Math.max(0, x - this.startPositionX);
|
|
||||||
}
|
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
private static class RecordTime implements Runnable {
|
private static class RecordTime implements Runnable {
|
||||||
|
|
||||||
private final TextView recordTimeView;
|
private final TextView recordTimeView;
|
||||||
private final AtomicLong startTime = new AtomicLong(0);
|
private final long limitSeconds;
|
||||||
|
private final Runnable onLimitHit;
|
||||||
|
private long startTime;
|
||||||
|
|
||||||
private RecordTime(TextView recordTimeView) {
|
private RecordTime(TextView recordTimeView, long limitSeconds, Runnable onLimitHit) {
|
||||||
this.recordTimeView = recordTimeView;
|
this.recordTimeView = recordTimeView;
|
||||||
|
this.limitSeconds = limitSeconds;
|
||||||
|
this.onLimitHit = onLimitHit;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@MainThread
|
||||||
public void display() {
|
public void display() {
|
||||||
this.startTime.set(System.currentTimeMillis());
|
this.startTime = System.currentTimeMillis();
|
||||||
this.recordTimeView.setText(DateUtils.formatElapsedTime(0));
|
this.recordTimeView.setText(DateUtils.formatElapsedTime(0));
|
||||||
ViewUtil.fadeIn(this.recordTimeView, FADE_TIME);
|
ViewUtil.fadeIn(this.recordTimeView, FADE_TIME);
|
||||||
Util.runOnMainDelayed(this, TimeUnit.SECONDS.toMillis(1));
|
Util.runOnMainDelayed(this, TimeUnit.SECONDS.toMillis(1));
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@MainThread
|
||||||
public long hide() {
|
public long hide() {
|
||||||
long elapsedtime = System.currentTimeMillis() - startTime.get();
|
long elapsedTime = System.currentTimeMillis() - startTime;
|
||||||
this.startTime.set(0);
|
this.startTime = 0;
|
||||||
ViewUtil.fadeOut(this.recordTimeView, FADE_TIME, View.INVISIBLE);
|
ViewUtil.fadeOut(this.recordTimeView, FADE_TIME, View.INVISIBLE);
|
||||||
return elapsedtime;
|
return elapsedTime;
|
||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
|
@MainThread
|
||||||
public void run() {
|
public void run() {
|
||||||
long localStartTime = startTime.get();
|
long localStartTime = startTime;
|
||||||
if (localStartTime > 0) {
|
if (localStartTime > 0) {
|
||||||
long elapsedTime = System.currentTimeMillis() - localStartTime;
|
long elapsedTime = System.currentTimeMillis() - localStartTime;
|
||||||
recordTimeView.setText(DateUtils.formatElapsedTime(TimeUnit.MILLISECONDS.toSeconds(elapsedTime)));
|
long elapsedSeconds = TimeUnit.MILLISECONDS.toSeconds(elapsedTime);
|
||||||
|
if (elapsedSeconds >= limitSeconds) {
|
||||||
|
onLimitHit.run();
|
||||||
|
} else {
|
||||||
|
recordTimeView.setText(DateUtils.formatElapsedTime(elapsedSeconds));
|
||||||
Util.runOnMainDelayed(this, TimeUnit.SECONDS.toMillis(1));
|
Util.runOnMainDelayed(this, TimeUnit.SECONDS.toMillis(1));
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
}
|
||||||
|
|
||||||
public interface MediaListener {
|
public interface MediaListener {
|
||||||
public void onMediaSelected(@NonNull Uri uri, String contentType);
|
void onMediaSelected(@NonNull Uri uri, String contentType);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -3,6 +3,7 @@ package org.thoughtcrime.securesms.components;
|
|||||||
import android.Manifest;
|
import android.Manifest;
|
||||||
import android.content.Context;
|
import android.content.Context;
|
||||||
import android.graphics.PorterDuff;
|
import android.graphics.PorterDuff;
|
||||||
|
import android.support.annotation.NonNull;
|
||||||
import android.support.annotation.Nullable;
|
import android.support.annotation.Nullable;
|
||||||
import android.support.v4.view.ViewCompat;
|
import android.support.v4.view.ViewCompat;
|
||||||
import android.util.AttributeSet;
|
import android.util.AttributeSet;
|
||||||
@ -12,6 +13,7 @@ import android.view.animation.Animation;
|
|||||||
import android.view.animation.AnimationSet;
|
import android.view.animation.AnimationSet;
|
||||||
import android.view.animation.AnticipateOvershootInterpolator;
|
import android.view.animation.AnticipateOvershootInterpolator;
|
||||||
import android.view.animation.DecelerateInterpolator;
|
import android.view.animation.DecelerateInterpolator;
|
||||||
|
import android.view.animation.LinearInterpolator;
|
||||||
import android.view.animation.OvershootInterpolator;
|
import android.view.animation.OvershootInterpolator;
|
||||||
import android.view.animation.ScaleAnimation;
|
import android.view.animation.ScaleAnimation;
|
||||||
import android.view.animation.TranslateAnimation;
|
import android.view.animation.TranslateAnimation;
|
||||||
@ -22,13 +24,20 @@ import org.thoughtcrime.securesms.R;
|
|||||||
import org.thoughtcrime.securesms.permissions.Permissions;
|
import org.thoughtcrime.securesms.permissions.Permissions;
|
||||||
import org.thoughtcrime.securesms.util.ViewUtil;
|
import org.thoughtcrime.securesms.util.ViewUtil;
|
||||||
|
|
||||||
public class MicrophoneRecorderView extends FrameLayout implements View.OnTouchListener {
|
public final class MicrophoneRecorderView extends FrameLayout implements View.OnTouchListener {
|
||||||
|
|
||||||
|
enum State {
|
||||||
|
NOT_RUNNING,
|
||||||
|
RUNNING_HELD,
|
||||||
|
RUNNING_LOCKED
|
||||||
|
}
|
||||||
|
|
||||||
public static final int ANIMATION_DURATION = 200;
|
public static final int ANIMATION_DURATION = 200;
|
||||||
|
|
||||||
private FloatingRecordButton floatingRecordButton;
|
private FloatingRecordButton floatingRecordButton;
|
||||||
|
private LockDropTarget lockDropTarget;
|
||||||
private @Nullable Listener listener;
|
private @Nullable Listener listener;
|
||||||
private boolean actionInProgress;
|
private @NonNull State state = State.NOT_RUNNING;
|
||||||
|
|
||||||
public MicrophoneRecorderView(Context context) {
|
public MicrophoneRecorderView(Context context) {
|
||||||
super(context);
|
super(context);
|
||||||
@ -42,22 +51,49 @@ public class MicrophoneRecorderView extends FrameLayout implements View.OnTouchL
|
|||||||
public void onFinishInflate() {
|
public void onFinishInflate() {
|
||||||
super.onFinishInflate();
|
super.onFinishInflate();
|
||||||
|
|
||||||
ImageView recordButtonFab = ViewUtil.findById(this, R.id.quick_audio_fab);
|
floatingRecordButton = new FloatingRecordButton(getContext(), findViewById(R.id.quick_audio_fab));
|
||||||
this.floatingRecordButton = new FloatingRecordButton(getContext(), recordButtonFab);
|
lockDropTarget = new LockDropTarget (getContext(), findViewById(R.id.lock_drop_target));
|
||||||
|
|
||||||
View recordButton = ViewUtil.findById(this, R.id.quick_audio_toggle);
|
View recordButton = ViewUtil.findById(this, R.id.quick_audio_toggle);
|
||||||
recordButton.setOnTouchListener(this);
|
recordButton.setOnTouchListener(this);
|
||||||
}
|
}
|
||||||
|
|
||||||
public void cancelAction() {
|
public void cancelAction() {
|
||||||
if (this.actionInProgress) {
|
if (state != State.NOT_RUNNING) {
|
||||||
this.actionInProgress = false;
|
state = State.NOT_RUNNING;
|
||||||
this.floatingRecordButton.hide(this.floatingRecordButton.lastPositionX);
|
hideUi();
|
||||||
|
|
||||||
if (listener != null) listener.onRecordCanceled(this.floatingRecordButton.lastPositionX);
|
if (listener != null) listener.onRecordCanceled();
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
public boolean isRecordingLocked() {
|
||||||
|
return state == State.RUNNING_LOCKED;
|
||||||
|
}
|
||||||
|
|
||||||
|
private void lockAction() {
|
||||||
|
if (state == State.RUNNING_HELD) {
|
||||||
|
state = State.RUNNING_LOCKED;
|
||||||
|
hideUi();
|
||||||
|
|
||||||
|
if (listener != null) listener.onRecordLocked();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
public void unlockAction() {
|
||||||
|
if (state == State.RUNNING_LOCKED) {
|
||||||
|
state = State.NOT_RUNNING;
|
||||||
|
hideUi();
|
||||||
|
|
||||||
|
if (listener != null) listener.onRecordReleased();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
private void hideUi() {
|
||||||
|
floatingRecordButton.hide();
|
||||||
|
lockDropTarget.hide();
|
||||||
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
public boolean onTouch(View v, final MotionEvent event) {
|
public boolean onTouch(View v, final MotionEvent event) {
|
||||||
switch (event.getAction()) {
|
switch (event.getAction()) {
|
||||||
@ -65,23 +101,29 @@ public class MicrophoneRecorderView extends FrameLayout implements View.OnTouchL
|
|||||||
if (!Permissions.hasAll(getContext(), Manifest.permission.RECORD_AUDIO)) {
|
if (!Permissions.hasAll(getContext(), Manifest.permission.RECORD_AUDIO)) {
|
||||||
if (listener != null) listener.onRecordPermissionRequired();
|
if (listener != null) listener.onRecordPermissionRequired();
|
||||||
} else {
|
} else {
|
||||||
this.actionInProgress = true;
|
state = State.RUNNING_HELD;
|
||||||
this.floatingRecordButton.display(event.getX());
|
floatingRecordButton.display(event.getX(), event.getY());
|
||||||
if (listener != null) listener.onRecordPressed(event.getX());
|
lockDropTarget.display();
|
||||||
|
if (listener != null) listener.onRecordPressed();
|
||||||
}
|
}
|
||||||
break;
|
break;
|
||||||
case MotionEvent.ACTION_CANCEL:
|
case MotionEvent.ACTION_CANCEL:
|
||||||
case MotionEvent.ACTION_UP:
|
case MotionEvent.ACTION_UP:
|
||||||
if (this.actionInProgress) {
|
if (this.state == State.RUNNING_HELD) {
|
||||||
this.actionInProgress = false;
|
state = State.NOT_RUNNING;
|
||||||
this.floatingRecordButton.hide(event.getX());
|
hideUi();
|
||||||
if (listener != null) listener.onRecordReleased(event.getX());
|
if (listener != null) listener.onRecordReleased();
|
||||||
}
|
}
|
||||||
break;
|
break;
|
||||||
case MotionEvent.ACTION_MOVE:
|
case MotionEvent.ACTION_MOVE:
|
||||||
if (this.actionInProgress) {
|
if (this.state == State.RUNNING_HELD) {
|
||||||
this.floatingRecordButton.moveTo(event.getX());
|
this.floatingRecordButton.moveTo(event.getX(), event.getY());
|
||||||
if (listener != null) listener.onRecordMoved(event.getX(), event.getRawX());
|
if (listener != null) listener.onRecordMoved(floatingRecordButton.lastOffsetX, event.getRawX());
|
||||||
|
|
||||||
|
int dimensionPixelSize = getResources().getDimensionPixelSize(R.dimen.recording_voice_lock_target);
|
||||||
|
if (floatingRecordButton.lastOffsetY <= dimensionPixelSize) {
|
||||||
|
lockAction();
|
||||||
|
}
|
||||||
}
|
}
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
@ -94,10 +136,11 @@ public class MicrophoneRecorderView extends FrameLayout implements View.OnTouchL
|
|||||||
}
|
}
|
||||||
|
|
||||||
public interface Listener {
|
public interface Listener {
|
||||||
void onRecordPressed(float x);
|
void onRecordPressed();
|
||||||
void onRecordReleased(float x);
|
void onRecordReleased();
|
||||||
void onRecordCanceled(float x);
|
void onRecordCanceled();
|
||||||
void onRecordMoved(float x, float absoluteX);
|
void onRecordLocked();
|
||||||
|
void onRecordMoved(float offsetX, float absoluteX);
|
||||||
void onRecordPermissionRequired();
|
void onRecordPermissionRequired();
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -106,81 +149,73 @@ public class MicrophoneRecorderView extends FrameLayout implements View.OnTouchL
|
|||||||
private final ImageView recordButtonFab;
|
private final ImageView recordButtonFab;
|
||||||
|
|
||||||
private float startPositionX;
|
private float startPositionX;
|
||||||
private float lastPositionX;
|
private float startPositionY;
|
||||||
|
private float lastOffsetX;
|
||||||
|
private float lastOffsetY;
|
||||||
|
|
||||||
public FloatingRecordButton(Context context, ImageView recordButtonFab) {
|
FloatingRecordButton(Context context, ImageView recordButtonFab) {
|
||||||
this.recordButtonFab = recordButtonFab;
|
this.recordButtonFab = recordButtonFab;
|
||||||
this.recordButtonFab.getBackground().setColorFilter(context.getResources()
|
this.recordButtonFab.getBackground().setColorFilter(context.getResources()
|
||||||
.getColor(R.color.red_500),
|
.getColor(R.color.red_500),
|
||||||
PorterDuff.Mode.SRC_IN);
|
PorterDuff.Mode.SRC_IN);
|
||||||
}
|
}
|
||||||
|
|
||||||
public void display(float x) {
|
void display(float x, float y) {
|
||||||
this.startPositionX = x;
|
this.startPositionX = x;
|
||||||
this.lastPositionX = x;
|
this.startPositionY = y;
|
||||||
|
|
||||||
recordButtonFab.setVisibility(View.VISIBLE);
|
recordButtonFab.setVisibility(View.VISIBLE);
|
||||||
|
|
||||||
float translation = ViewCompat.getLayoutDirection(recordButtonFab) ==
|
|
||||||
ViewCompat.LAYOUT_DIRECTION_LTR ? -.25f : .25f;
|
|
||||||
|
|
||||||
AnimationSet animation = new AnimationSet(true);
|
AnimationSet animation = new AnimationSet(true);
|
||||||
animation.addAnimation(new TranslateAnimation(Animation.RELATIVE_TO_SELF, 0,
|
animation.addAnimation(new TranslateAnimation(Animation.ABSOLUTE, 0,
|
||||||
Animation.RELATIVE_TO_SELF, 0,
|
Animation.ABSOLUTE, 0,
|
||||||
Animation.RELATIVE_TO_SELF, -.25f,
|
Animation.ABSOLUTE, 0,
|
||||||
Animation.RELATIVE_TO_SELF, -.25f));
|
Animation.ABSOLUTE, 0));
|
||||||
|
|
||||||
animation.addAnimation(new ScaleAnimation(.5f, 1f, .5f, 1f,
|
animation.addAnimation(new ScaleAnimation(.5f, 1f, .5f, 1f,
|
||||||
Animation.RELATIVE_TO_SELF, .5f,
|
Animation.RELATIVE_TO_SELF, .5f,
|
||||||
Animation.RELATIVE_TO_SELF, .5f));
|
Animation.RELATIVE_TO_SELF, .5f));
|
||||||
|
|
||||||
animation.setFillBefore(true);
|
|
||||||
animation.setFillAfter(true);
|
|
||||||
animation.setDuration(ANIMATION_DURATION);
|
animation.setDuration(ANIMATION_DURATION);
|
||||||
animation.setInterpolator(new OvershootInterpolator());
|
animation.setInterpolator(new OvershootInterpolator());
|
||||||
|
|
||||||
recordButtonFab.startAnimation(animation);
|
recordButtonFab.startAnimation(animation);
|
||||||
}
|
}
|
||||||
|
|
||||||
public void moveTo(float x) {
|
void moveTo(float x, float y) {
|
||||||
this.lastPositionX = x;
|
lastOffsetX = getXOffset(x);
|
||||||
|
lastOffsetY = getYOffset(y);
|
||||||
|
|
||||||
float offset = getOffset(x);
|
if (Math.abs(lastOffsetX) > Math.abs(lastOffsetY)) {
|
||||||
|
lastOffsetY = 0;
|
||||||
Animation translateAnimation = new TranslateAnimation(Animation.ABSOLUTE, offset,
|
} else {
|
||||||
Animation.ABSOLUTE, offset,
|
lastOffsetX = 0;
|
||||||
Animation.RELATIVE_TO_SELF, -.25f,
|
|
||||||
Animation.RELATIVE_TO_SELF, -.25f);
|
|
||||||
|
|
||||||
translateAnimation.setDuration(0);
|
|
||||||
translateAnimation.setFillAfter(true);
|
|
||||||
translateAnimation.setFillBefore(true);
|
|
||||||
|
|
||||||
recordButtonFab.startAnimation(translateAnimation);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
public void hide(float x) {
|
recordButtonFab.setTranslationX(lastOffsetX);
|
||||||
this.lastPositionX = x;
|
recordButtonFab.setTranslationY(lastOffsetY);
|
||||||
|
}
|
||||||
|
|
||||||
float offset = getOffset(x);
|
void hide() {
|
||||||
|
recordButtonFab.setTranslationX(0);
|
||||||
|
recordButtonFab.setTranslationY(0);
|
||||||
|
if (recordButtonFab.getVisibility() != VISIBLE) return;
|
||||||
|
|
||||||
AnimationSet animation = new AnimationSet(false);
|
AnimationSet animation = new AnimationSet(false);
|
||||||
Animation scaleAnimation = new ScaleAnimation(1, 0.5f, 1, 0.5f,
|
Animation scaleAnimation = new ScaleAnimation(1, 0.5f, 1, 0.5f,
|
||||||
Animation.RELATIVE_TO_SELF, 0.5f,
|
Animation.RELATIVE_TO_SELF, 0.5f,
|
||||||
Animation.RELATIVE_TO_SELF, 0.5f);
|
Animation.RELATIVE_TO_SELF, 0.5f);
|
||||||
|
|
||||||
Animation translateAnimation = new TranslateAnimation(Animation.ABSOLUTE, offset,
|
Animation translateAnimation = new TranslateAnimation(Animation.ABSOLUTE, lastOffsetX,
|
||||||
Animation.ABSOLUTE, 0,
|
Animation.ABSOLUTE, 0,
|
||||||
Animation.RELATIVE_TO_SELF, -.25f,
|
Animation.ABSOLUTE, lastOffsetY,
|
||||||
Animation.RELATIVE_TO_SELF, -.25f);
|
Animation.ABSOLUTE, 0);
|
||||||
|
|
||||||
scaleAnimation.setInterpolator(new AnticipateOvershootInterpolator(1.5f));
|
scaleAnimation.setInterpolator(new AnticipateOvershootInterpolator(1.5f));
|
||||||
translateAnimation.setInterpolator(new DecelerateInterpolator());
|
translateAnimation.setInterpolator(new DecelerateInterpolator());
|
||||||
animation.addAnimation(scaleAnimation);
|
animation.addAnimation(scaleAnimation);
|
||||||
animation.addAnimation(translateAnimation);
|
animation.addAnimation(translateAnimation);
|
||||||
animation.setDuration(ANIMATION_DURATION);
|
animation.setDuration(ANIMATION_DURATION);
|
||||||
animation.setFillBefore(true);
|
|
||||||
animation.setFillAfter(false);
|
|
||||||
animation.setInterpolator(new AnticipateOvershootInterpolator(1.5f));
|
animation.setInterpolator(new AnticipateOvershootInterpolator(1.5f));
|
||||||
|
|
||||||
recordButtonFab.setVisibility(View.GONE);
|
recordButtonFab.setVisibility(View.GONE);
|
||||||
@ -188,16 +223,48 @@ public class MicrophoneRecorderView extends FrameLayout implements View.OnTouchL
|
|||||||
recordButtonFab.startAnimation(animation);
|
recordButtonFab.startAnimation(animation);
|
||||||
}
|
}
|
||||||
|
|
||||||
private float getOffset(float x) {
|
private float getXOffset(float x) {
|
||||||
return ViewCompat.getLayoutDirection(recordButtonFab) == ViewCompat.LAYOUT_DIRECTION_LTR ?
|
return ViewCompat.getLayoutDirection(recordButtonFab) == ViewCompat.LAYOUT_DIRECTION_LTR ?
|
||||||
-Math.max(0, this.startPositionX - x) : Math.max(0, x - this.startPositionX);
|
-Math.max(0, this.startPositionX - x) : Math.max(0, x - this.startPositionX);
|
||||||
}
|
}
|
||||||
|
|
||||||
private int getWidthAdjustment() {
|
private float getYOffset(float y) {
|
||||||
int width = recordButtonFab.getWidth() / 4;
|
return Math.min(0, y - this.startPositionY);
|
||||||
return ViewCompat.getLayoutDirection(recordButtonFab) == ViewCompat.LAYOUT_DIRECTION_LTR ? -width : width;
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
private static class LockDropTarget {
|
||||||
|
|
||||||
|
private final View lockDropTarget;
|
||||||
|
private final int dropTargetPosition;
|
||||||
|
|
||||||
|
LockDropTarget(Context context, View lockDropTarget) {
|
||||||
|
this.lockDropTarget = lockDropTarget;
|
||||||
|
this.dropTargetPosition = context.getResources().getDimensionPixelSize(R.dimen.recording_voice_lock_target);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
void display() {
|
||||||
|
lockDropTarget.setScaleX(1);
|
||||||
|
lockDropTarget.setScaleY(1);
|
||||||
|
lockDropTarget.setAlpha(0);
|
||||||
|
lockDropTarget.setTranslationY(0);
|
||||||
|
lockDropTarget.setVisibility(VISIBLE);
|
||||||
|
lockDropTarget.animate()
|
||||||
|
.setStartDelay(ANIMATION_DURATION * 2)
|
||||||
|
.setDuration(ANIMATION_DURATION)
|
||||||
|
.setInterpolator(new DecelerateInterpolator())
|
||||||
|
.translationY(dropTargetPosition)
|
||||||
|
.alpha(1)
|
||||||
|
.start();
|
||||||
|
}
|
||||||
|
|
||||||
|
void hide() {
|
||||||
|
lockDropTarget.animate()
|
||||||
|
.setStartDelay(0)
|
||||||
|
.setDuration(ANIMATION_DURATION)
|
||||||
|
.setInterpolator(new LinearInterpolator())
|
||||||
|
.scaleX(0).scaleY(0)
|
||||||
|
.start();
|
||||||
|
}
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
@ -45,6 +45,8 @@ public class SendButton extends ImageButton
|
|||||||
}
|
}
|
||||||
|
|
||||||
private TransportOptions initializeTransportOptions(boolean media) {
|
private TransportOptions initializeTransportOptions(boolean media) {
|
||||||
|
if (isInEditMode()) return null;
|
||||||
|
|
||||||
TransportOptions transportOptions = new TransportOptions(getContext(), media);
|
TransportOptions transportOptions = new TransportOptions(getContext(), media);
|
||||||
transportOptions.addOnTransportChangedListener(this);
|
transportOptions.addOnTransportChangedListener(this);
|
||||||
|
|
||||||
|
@ -1995,6 +1995,11 @@ public class ConversationActivity extends PassphraseRequiredActionBarActivity
|
|||||||
}
|
}
|
||||||
|
|
||||||
private void sendMessage() {
|
private void sendMessage() {
|
||||||
|
if (inputPanel.isRecordingInLockedMode()) {
|
||||||
|
inputPanel.releaseRecordingLock();
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
try {
|
try {
|
||||||
Recipient recipient = getRecipient();
|
Recipient recipient = getRecipient();
|
||||||
|
|
||||||
@ -2176,6 +2181,13 @@ public class ConversationActivity extends PassphraseRequiredActionBarActivity
|
|||||||
}
|
}
|
||||||
|
|
||||||
private void updateToggleButtonState() {
|
private void updateToggleButtonState() {
|
||||||
|
if (inputPanel.isRecordingInLockedMode()) {
|
||||||
|
buttonToggle.display(sendButton);
|
||||||
|
quickAttachmentToggle.show();
|
||||||
|
inlineAttachmentToggle.hide();
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
if (composeText.getText().length() == 0 && !attachmentManager.isAttachmentPresent()) {
|
if (composeText.getText().length() == 0 && !attachmentManager.isAttachmentPresent()) {
|
||||||
buttonToggle.display(attachButton);
|
buttonToggle.display(attachButton);
|
||||||
quickAttachmentToggle.show();
|
quickAttachmentToggle.show();
|
||||||
@ -2232,8 +2244,14 @@ public class ConversationActivity extends PassphraseRequiredActionBarActivity
|
|||||||
audioRecorder.startRecording();
|
audioRecorder.startRecording();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public void onRecorderLocked() {
|
||||||
|
updateToggleButtonState();
|
||||||
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
public void onRecorderFinished() {
|
public void onRecorderFinished() {
|
||||||
|
updateToggleButtonState();
|
||||||
Vibrator vibrator = ServiceUtil.getVibrator(this);
|
Vibrator vibrator = ServiceUtil.getVibrator(this);
|
||||||
vibrator.vibrate(20);
|
vibrator.vibrate(20);
|
||||||
|
|
||||||
@ -2274,6 +2292,7 @@ public class ConversationActivity extends PassphraseRequiredActionBarActivity
|
|||||||
|
|
||||||
@Override
|
@Override
|
||||||
public void onRecorderCanceled() {
|
public void onRecorderCanceled() {
|
||||||
|
updateToggleButtonState();
|
||||||
Vibrator vibrator = ServiceUtil.getVibrator(this);
|
Vibrator vibrator = ServiceUtil.getVibrator(this);
|
||||||
vibrator.vibrate(50);
|
vibrator.vibrate(50);
|
||||||
|
|
||||||
|
Loading…
x
Reference in New Issue
Block a user