Skip to content

Commit

Permalink
Transcript display using dialog
Browse files Browse the repository at this point in the history
  • Loading branch information
tonytamsf authored and ByteHamster committed Apr 15, 2024
1 parent 1edceb1 commit 57b4e39
Show file tree
Hide file tree
Showing 18 changed files with 648 additions and 28 deletions.
1 change: 1 addition & 0 deletions app/build.gradle
Expand Up @@ -69,6 +69,7 @@ dependencies {
implementation project(':net:ssl')
implementation project(':net:sync:service')
implementation project(':parser:feed')
implementation project(':parser:transcript')
implementation project(':playback:base')
implementation project(':playback:cast')
implementation project(':storage:database')
Expand Down
Expand Up @@ -60,6 +60,7 @@ public static boolean onPrepareMenu(Menu menu, FeedItem selectedItem) {
final boolean fileDownloaded = hasMedia && selectedItem.getMedia().fileExists();
final boolean isLocalFile = hasMedia && selectedItem.getFeed().isLocalFeed();
final boolean isFavorite = selectedItem.isTagged(FeedItem.TAG_FAVORITE);
final boolean hasTranscript = selectedItem.hasTranscript();

setItemVisibility(menu, R.id.skip_episode_item, isPlaying);
setItemVisibility(menu, R.id.remove_from_queue_item, isInQueue);
Expand All @@ -84,6 +85,7 @@ public static boolean onPrepareMenu(Menu menu, FeedItem selectedItem) {
setItemVisibility(menu, R.id.add_to_favorites_item, !isFavorite);
setItemVisibility(menu, R.id.remove_from_favorites_item, isFavorite);
setItemVisibility(menu, R.id.remove_item, fileDownloaded || isLocalFile);
setItemVisibility(menu, R.id.transcript_item, hasTranscript);
return true;
}

Expand Down
@@ -0,0 +1,159 @@
package de.danoeh.antennapod.ui.screen.playback;

import androidx.recyclerview.widget.RecyclerView;

import android.util.Log;
import android.view.LayoutInflater;
import android.view.View;
import android.view.ViewGroup;
import android.widget.TextView;

import de.danoeh.antennapod.playback.service.PlaybackController;
import org.greenrobot.eventbus.Subscribe;
import org.greenrobot.eventbus.ThreadMode;
import org.jsoup.internal.StringUtil;

import de.danoeh.antennapod.databinding.FragmentItemTranscriptRvBinding;
import de.danoeh.antennapod.event.playback.PlaybackPositionEvent;
import de.danoeh.antennapod.model.feed.Transcript;
import de.danoeh.antennapod.model.feed.TranscriptSegment;
import de.danoeh.antennapod.parser.transcript.TranscriptParser;
import de.danoeh.antennapod.playback.base.PlayerStatus;

import java.util.Hashtable;
import java.util.Map;
import java.util.SortedMap;
import java.util.TreeMap;

/**
* {@link RecyclerView.Adapter} that can display a {@link PlaceholderItem}.
* TODO: Replace the implementation with code for your data type.
*/
public class ItemTranscriptRvAdapter extends RecyclerView.Adapter<ItemTranscriptRvAdapter.ViewHolder> {

public String tag = "ItemTranscriptRVAdapter";
public Hashtable<Long, Integer> positions;
public Hashtable<Integer, TranscriptSegment> snippets;
PlaybackController controller;

private Transcript transcript;

public ItemTranscriptRvAdapter(Transcript t) {
positions = new Hashtable<Long, Integer>();
snippets = new Hashtable<Integer, TranscriptSegment>();
setTranscript(t);
}

@Override
public ViewHolder onCreateViewHolder(ViewGroup parent, int viewType) {

return new ViewHolder(FragmentItemTranscriptRvBinding.inflate(LayoutInflater.from(parent.getContext()),
parent,
false));

}

public void setController(PlaybackController controller) {
this.controller = controller;
}

public void setTranscript(Transcript t) {
transcript = t;
if (transcript == null) {
return;
}
TreeMap<Long, TranscriptSegment> segmentsMap = transcript.getSegmentsMap();
Object[] objs = segmentsMap.entrySet().toArray();
for (int i = 0; i < objs.length; i++) {
Map.Entry<Long, TranscriptSegment> seg;
seg = (Map.Entry<Long, TranscriptSegment>) objs[i];
positions.put((Long) seg.getKey(), i);
snippets.put(i, seg.getValue());
}
}

@Override
public void onBindViewHolder(final ViewHolder holder, int position) {
TreeMap<Long, TranscriptSegment> segmentsMap;
SortedMap<Long, TranscriptSegment> map;

segmentsMap = transcript.getSegmentsMap();
// TODO: fix this performance
TreeMap.Entry entry = (TreeMap.Entry) segmentsMap.entrySet().toArray()[position];
TranscriptSegment seg = (TranscriptSegment) entry.getValue();
Long k = (Long) entry.getKey();

Log.d(tag, "onBindViewHolder position " + position + " RV pos " + k);
holder.transcriptSegment = seg;
holder.viewTimecode.setText(TranscriptParser.secondsToTime(k));
holder.viewTimecode.setVisibility(View.GONE);
if (! StringUtil.isBlank(seg.getSpeaker())) {
TreeMap.Entry prevEntry = null;
try {
prevEntry = (TreeMap.Entry) segmentsMap.entrySet().toArray()[position - 1];
} catch (ArrayIndexOutOfBoundsException e) {
Log.d(tag, "ArrayIndexOutOfBoundsException");
}
TranscriptSegment prevSeg = null;
if (prevEntry != null) {
prevSeg = (TranscriptSegment) prevEntry.getValue();
}
if (prevEntry != null && prevSeg.getSpeaker().equals(seg.getSpeaker())) {
holder.viewTimecode.setVisibility(View.GONE);
holder.viewContent.setText(seg.getWords());
} else {
holder.viewTimecode.setText(TranscriptParser.secondsToTime(k) + " " + seg.getSpeaker());
holder.viewContent.setText(seg.getWords());
holder.viewTimecode.setVisibility(View.VISIBLE);
}
} else {
holder.viewContent.setText(seg.getWords());
}
}

@Subscribe(threadMode = ThreadMode.MAIN)
public void onEventMainThread(PlaybackPositionEvent event) {
Log.d(tag, "onEventMainThread ItemTranscriptRVAdapter");
}

@Override
public int getItemCount() {
if (transcript == null) {
return 0;
}
return transcript.getSegmentsMap().size();
}

public class ViewHolder extends RecyclerView.ViewHolder {
public final TextView viewTimecode;
public final TextView viewContent;
public TranscriptSegment transcriptSegment;

public ViewHolder(FragmentItemTranscriptRvBinding binding) {
super(binding.getRoot());
viewTimecode = binding.speaker;
viewContent = binding.content;
viewContent.setOnClickListener(v -> {
Log.d(tag, "Clicked on " + transcriptSegment.getWords());
long startTime = transcriptSegment.getStartTime();
long endTime = transcriptSegment.getEndTime();
if (! (controller.getPosition() >= startTime
&& controller.getPosition() <= endTime)) {
controller.seekTo((int) startTime);

if (controller.getStatus() == PlayerStatus.PAUSED
|| controller.getStatus() == PlayerStatus.STOPPED) {
controller.playPause();
}
} else {
controller.playPause();
}
});
}

@Override
public String toString() {
return super.toString() + " '" + viewContent.getText() + "'";
}
}
}

0 comments on commit 57b4e39

Please sign in to comment.