Working on documentation. Table rendering sample
This commit is contained in:
parent
cc75a92c7f
commit
96ca96fa70
@ -1 +1,125 @@
|
||||
# Images
|
||||
|
||||
Starting with <Badge text="3.0.0" /> `Markwon` comes with `ImagesPlugin`
|
||||
which supports `http(s)`, `file` and `data` schemes and default media
|
||||
decoder (for simple images, no [SVG](/docs/image/svg.md) or [GIF](/docs/image/gif.md) which
|
||||
are defined in standalone modules).
|
||||
|
||||
## ImagesPlugin
|
||||
|
||||
`ImagePlugin` takes care of _obtaining_ image resource, decoding it and displaying it in a `TextView`.
|
||||
|
||||
:::warning
|
||||
Although `core` artifact contains `ImagesPlugin` one must
|
||||
still **explicitly** register the `ImagesPlugin` on resulting `Markwon`
|
||||
instance.
|
||||
```java
|
||||
final Markwon markwon = Markwon.builder(context)
|
||||
.usePlugin(ImagesPlugin.create())
|
||||
```
|
||||
:::
|
||||
|
||||
There are 2 factory methods to obtain `ImagesPlugin`:
|
||||
* `ImagesPlugin#create(Context)`
|
||||
* `ImagesPlugin#createWithAssets(Context)`
|
||||
|
||||
The first one `#create(Context)` configures:
|
||||
* `FileSchemeHandler` that allows obtaining images from `file://` uris
|
||||
* `DataUriSchemeHandler` that allows _inlining_ images with `data:`
|
||||
scheme (``)
|
||||
* `NetworkSchemeHandler` that allows obtaining images from `http://` and `https://` uris
|
||||
(internally it uses `HttpURLConnection`)
|
||||
* `ImageMediaDecoder` which _tries_ to decode all encountered images as regular ones (png, jpg, etc)
|
||||
|
||||
The second one `#createWithAssets(Context)` does the same but also adds support
|
||||
for images that reside in `assets` folder of your application and
|
||||
referenced by `file:///android_asset/{path}` uri.
|
||||
|
||||
`ImagesPlugin` also _prepares_ a TextView to display images. Due to asynchronous
|
||||
nature of image loading, there must be a way to invalidate resulting Spanned
|
||||
content after an image is loaded.
|
||||
|
||||
:::warning
|
||||
Images come with few limitations. For of all, they work with a **TextView only**.
|
||||
This is due to the fact that there is no way to invalidate a `Spanned` content
|
||||
by itself (without context in which it is displayed). So, if `Markwon` is used,
|
||||
for example, to display a `Toast` with an image:
|
||||
|
||||
```java
|
||||
final Spanned spanned = markwon.toMarkdown("Hello ");
|
||||
Toast.makeText(context, spanned, Toast.LENGTH_LONG).show();
|
||||
```
|
||||
|
||||
Image _probably_ won't be displayed. As a workaround for `Toast` a custom `View`
|
||||
can be used:
|
||||
|
||||
```java
|
||||
final Spanned spanned = markwon.toMarkdown("Hello ");
|
||||
|
||||
final View view = createToastView();
|
||||
final TextView textView = view.findViewById(R.id.text_view);
|
||||
markwon.setParsedMarkdown(textView, spanned);
|
||||
|
||||
final Toast toast = new Toast(context);
|
||||
toast.setView(view);
|
||||
// other Toast configurations
|
||||
toast.show();
|
||||
```
|
||||
:::
|
||||
|
||||
## SchemeHandler
|
||||
|
||||
To add support for different schemes (or customize provided) a `SchemeHandler` must be used.
|
||||
|
||||
```java
|
||||
final Markwon markwon = Markwon.builder(context)
|
||||
.usePlugin(ImagesPlugin.create(context))
|
||||
.usePlugin(new AbstractMarkwonPlugin() {
|
||||
@Override
|
||||
public void configureImages(@NonNull AsyncDrawableLoader.Builder builder) {
|
||||
// example only, Markwon doesn't come with a ftp scheme handler
|
||||
builder.addSchemeHandler("ftp", new FtpSchemeHandler());
|
||||
}
|
||||
})
|
||||
.build();
|
||||
```
|
||||
|
||||
It's a class to _convert_ an URI into an `InputStream`:
|
||||
|
||||
```java
|
||||
public abstract class SchemeHandler {
|
||||
|
||||
@Nullable
|
||||
public abstract ImageItem handle(@NonNull String raw, @NonNull Uri uri);
|
||||
}
|
||||
```
|
||||
|
||||
`ImageItem` is a holder class for resulting `InputStream` and (optional)
|
||||
content type:
|
||||
|
||||
```java
|
||||
public class ImageItem {
|
||||
|
||||
private final String contentType;
|
||||
private final InputStream inputStream;
|
||||
|
||||
/* rest omitted */
|
||||
}
|
||||
```
|
||||
|
||||
Based on `contentType` returned a corresponding `MediaDecoder` will be matched.
|
||||
If no `MediaDecoder` can handle given `contentType` then a default media decoder will
|
||||
be used.
|
||||
|
||||
## MediaDecoder
|
||||
|
||||
:::tip
|
||||
If you are using [html](/docs/html/) you do not have to additionally setup
|
||||
images displayed via `<img>` tag, as `HtmlPlugin` automatically uses configured
|
||||
image loader. But images referenced in HTML come with additional support for
|
||||
sizes, which is not supported natively by markdown, allowing absolute or relative sizes:
|
||||
|
||||
```html
|
||||
<img src="./assets/my-image" width="100%">
|
||||
```
|
||||
:::
|
@ -2,6 +2,23 @@
|
||||
|
||||
Here is the list of properties that can be configured via `MarkwonTheme.Builder` class.
|
||||
|
||||
:::tip
|
||||
Starting with <Badge text="3.0.0" /> there is no need to manually construct a `MarkwonTheme`.
|
||||
Instead a `Plugin` should be used:
|
||||
```java
|
||||
final Markwon markwon = Markwon.builder(context)
|
||||
.usePlugin(new AbstractMarkwonPlugin() {
|
||||
@Override
|
||||
public void configureTheme(@NonNull MarkwonTheme.Builder builder) {
|
||||
builder
|
||||
.codeTextColor(Color.BLACK)
|
||||
.codeBackgroundColor(Color.GREEN);
|
||||
}
|
||||
})
|
||||
.build();
|
||||
```
|
||||
:::
|
||||
|
||||
## Link color
|
||||
|
||||
Controls the color of a [link](#)
|
||||
|
@ -10,7 +10,6 @@ import org.commonmark.ext.gfm.tables.TableHead;
|
||||
import org.commonmark.ext.gfm.tables.TableRow;
|
||||
import org.commonmark.node.AbstractVisitor;
|
||||
import org.commonmark.node.CustomNode;
|
||||
import org.commonmark.node.Node;
|
||||
|
||||
import java.util.ArrayList;
|
||||
import java.util.List;
|
||||
@ -160,20 +159,12 @@ public class Table {
|
||||
|
||||
final TableCell cell = (TableCell) customNode;
|
||||
|
||||
final Node firstChild = cell.getFirstChild();
|
||||
|
||||
// need to investigate why... (most likely initial node is modified by someone)
|
||||
if (firstChild != null) {
|
||||
|
||||
if (pendingRow == null) {
|
||||
pendingRow = new ArrayList<>(2);
|
||||
}
|
||||
|
||||
// let's TRY to not visit this node but instead try to render its first child
|
||||
|
||||
pendingRow.add(new Table.Column(alignment(cell.getAlignment()), markwon.render(firstChild)));
|
||||
pendingRow.add(new Table.Column(alignment(cell.getAlignment()), markwon.render(cell)));
|
||||
pendingRowIsHeader = cell.isHeader();
|
||||
}
|
||||
|
||||
return;
|
||||
}
|
||||
@ -215,6 +206,4 @@ public class Table {
|
||||
return out;
|
||||
}
|
||||
}
|
||||
|
||||
|
||||
}
|
||||
|
@ -67,7 +67,7 @@ public abstract class MarkwonAdapter extends RecyclerView.Adapter<MarkwonAdapter
|
||||
* @see SimpleEntry
|
||||
*/
|
||||
@NonNull
|
||||
public static MarkwonAdapter create(@NonNull Entry<? extends Holder, ? extends Node> defaultEntry) {
|
||||
public static MarkwonAdapter create(@NonNull Entry<? extends Node, ? extends Holder> defaultEntry) {
|
||||
return new MarkwonAdapterImpl.BuilderImpl().defaultEntry(defaultEntry).build();
|
||||
}
|
||||
|
||||
@ -109,7 +109,7 @@ public abstract class MarkwonAdapter extends RecyclerView.Adapter<MarkwonAdapter
|
||||
@NonNull
|
||||
<N extends Node> Builder include(
|
||||
@NonNull Class<N> node,
|
||||
@NonNull Entry<? extends Holder, ? super N> entry);
|
||||
@NonNull Entry<? super N, ? extends Holder> entry);
|
||||
|
||||
/**
|
||||
* Specify which {@link Entry} to use for all non-explicitly registered nodes
|
||||
@ -119,7 +119,7 @@ public abstract class MarkwonAdapter extends RecyclerView.Adapter<MarkwonAdapter
|
||||
* @see SimpleEntry
|
||||
*/
|
||||
@NonNull
|
||||
Builder defaultEntry(@NonNull Entry<? extends Holder, ? extends Node> defaultEntry);
|
||||
Builder defaultEntry(@NonNull Entry<? extends Node, ? extends Holder> defaultEntry);
|
||||
|
||||
/**
|
||||
* Specify which layout {@link SimpleEntry} will use to render all non-explicitly
|
||||
@ -156,7 +156,7 @@ public abstract class MarkwonAdapter extends RecyclerView.Adapter<MarkwonAdapter
|
||||
/**
|
||||
* @see SimpleEntry
|
||||
*/
|
||||
public interface Entry<H extends Holder, N extends Node> {
|
||||
public interface Entry<N extends Node, H extends Holder> {
|
||||
|
||||
@NonNull
|
||||
H createHolder(@NonNull LayoutInflater inflater, @NonNull ViewGroup parent);
|
||||
|
@ -15,8 +15,8 @@ import ru.noties.markwon.Markwon;
|
||||
|
||||
class MarkwonAdapterImpl extends MarkwonAdapter {
|
||||
|
||||
private final SparseArray<Entry<Holder, Node>> entries;
|
||||
private final Entry<Holder, Node> defaultEntry;
|
||||
private final SparseArray<Entry<Node, Holder>> entries;
|
||||
private final Entry<Node, Holder> defaultEntry;
|
||||
private final Reducer reducer;
|
||||
|
||||
private LayoutInflater layoutInflater;
|
||||
@ -26,8 +26,8 @@ class MarkwonAdapterImpl extends MarkwonAdapter {
|
||||
|
||||
@SuppressWarnings("WeakerAccess")
|
||||
MarkwonAdapterImpl(
|
||||
@NonNull SparseArray<Entry<Holder, Node>> entries,
|
||||
@NonNull Entry<Holder, Node> defaultEntry,
|
||||
@NonNull SparseArray<Entry<Node, Holder>> entries,
|
||||
@NonNull Entry<Node, Holder> defaultEntry,
|
||||
@NonNull Reducer reducer) {
|
||||
this.entries = entries;
|
||||
this.defaultEntry = defaultEntry;
|
||||
@ -67,7 +67,7 @@ class MarkwonAdapterImpl extends MarkwonAdapter {
|
||||
layoutInflater = LayoutInflater.from(parent.getContext());
|
||||
}
|
||||
|
||||
final Entry<Holder, Node> entry = getEntry(viewType);
|
||||
final Entry<Node, Holder> entry = getEntry(viewType);
|
||||
|
||||
return entry.createHolder(layoutInflater, parent);
|
||||
}
|
||||
@ -78,7 +78,7 @@ class MarkwonAdapterImpl extends MarkwonAdapter {
|
||||
final Node node = nodes.get(position);
|
||||
final int viewType = getNodeViewType(node.getClass());
|
||||
|
||||
final Entry<Holder, Node> entry = getEntry(viewType);
|
||||
final Entry<Node, Holder> entry = getEntry(viewType);
|
||||
|
||||
entry.bindHolder(markwon, holder, node);
|
||||
}
|
||||
@ -107,7 +107,7 @@ class MarkwonAdapterImpl extends MarkwonAdapter {
|
||||
public long getItemId(int position) {
|
||||
final Node node = nodes.get(position);
|
||||
final int type = getNodeViewType(node.getClass());
|
||||
final Entry<Holder, Node> entry = getEntry(type);
|
||||
final Entry<Node, Holder> entry = getEntry(type);
|
||||
return entry.id(node);
|
||||
}
|
||||
|
||||
@ -122,7 +122,7 @@ class MarkwonAdapterImpl extends MarkwonAdapter {
|
||||
}
|
||||
|
||||
@NonNull
|
||||
private Entry<Holder, Node> getEntry(int viewType) {
|
||||
private Entry<Node, Holder> getEntry(int viewType) {
|
||||
return viewType == 0
|
||||
? defaultEntry
|
||||
: entries.get(viewType);
|
||||
@ -130,26 +130,26 @@ class MarkwonAdapterImpl extends MarkwonAdapter {
|
||||
|
||||
static class BuilderImpl implements Builder {
|
||||
|
||||
private final SparseArray<Entry<Holder, Node>> entries = new SparseArray<>(3);
|
||||
private final SparseArray<Entry<Node, Holder>> entries = new SparseArray<>(3);
|
||||
|
||||
private Entry<Holder, Node> defaultEntry;
|
||||
private Entry<Node, Holder> defaultEntry;
|
||||
private Reducer reducer;
|
||||
|
||||
@NonNull
|
||||
@Override
|
||||
public <N extends Node> Builder include(
|
||||
@NonNull Class<N> node,
|
||||
@NonNull Entry<? extends Holder, ? super N> entry) {
|
||||
@NonNull Entry<? super N, ? extends Holder> entry) {
|
||||
//noinspection unchecked
|
||||
entries.append(node.hashCode(), (Entry<Holder, Node>) entry);
|
||||
entries.append(node.hashCode(), (Entry<Node, Holder>) entry);
|
||||
return this;
|
||||
}
|
||||
|
||||
@NonNull
|
||||
@Override
|
||||
public Builder defaultEntry(@NonNull Entry<? extends Holder, ? extends Node> defaultEntry) {
|
||||
public Builder defaultEntry(@NonNull Entry<? extends Node, ? extends Holder> defaultEntry) {
|
||||
//noinspection unchecked
|
||||
this.defaultEntry = (Entry<Holder, Node>) defaultEntry;
|
||||
this.defaultEntry = (Entry<Node, Holder>) defaultEntry;
|
||||
return this;
|
||||
}
|
||||
|
||||
@ -157,7 +157,7 @@ class MarkwonAdapterImpl extends MarkwonAdapter {
|
||||
@Override
|
||||
public Builder defaultEntry(int layoutResId) {
|
||||
//noinspection unchecked
|
||||
this.defaultEntry = (Entry<Holder, Node>) (Entry) new SimpleEntry(layoutResId);
|
||||
this.defaultEntry = (Entry<Node, Holder>) (Entry) new SimpleEntry(layoutResId);
|
||||
return this;
|
||||
}
|
||||
|
||||
@ -174,7 +174,7 @@ class MarkwonAdapterImpl extends MarkwonAdapter {
|
||||
|
||||
if (defaultEntry == null) {
|
||||
//noinspection unchecked
|
||||
defaultEntry = (Entry<Holder, Node>) (Entry) new SimpleEntry();
|
||||
defaultEntry = (Entry<Node, Holder>) (Entry) new SimpleEntry();
|
||||
}
|
||||
|
||||
if (reducer == null) {
|
||||
|
@ -21,7 +21,7 @@ import ru.noties.markwon.Markwon;
|
||||
* @since 3.0.0
|
||||
*/
|
||||
@SuppressWarnings("WeakerAccess")
|
||||
public class SimpleEntry implements MarkwonAdapter.Entry<SimpleEntry.Holder, Node> {
|
||||
public class SimpleEntry implements MarkwonAdapter.Entry<Node, SimpleEntry.Holder> {
|
||||
|
||||
public static final Spannable.Factory NO_COPY_SPANNABLE_FACTORY = new NoCopySpannableFactory();
|
||||
|
||||
|
@ -8,8 +8,8 @@
|
||||
<application
|
||||
android:allowBackup="true"
|
||||
android:icon="@mipmap/ic_launcher"
|
||||
android:roundIcon="@mipmap/ic_launcher"
|
||||
android:label="@string/app_name"
|
||||
android:roundIcon="@mipmap/ic_launcher"
|
||||
android:theme="@style/AppTheme"
|
||||
tools:ignore="AllowBackup,GoogleAppIndexingWarning">
|
||||
|
||||
|
@ -11,27 +11,29 @@ import android.support.v7.widget.RecyclerView;
|
||||
import android.text.TextUtils;
|
||||
|
||||
import org.commonmark.ext.gfm.tables.TableBlock;
|
||||
import org.commonmark.ext.gfm.tables.TablesExtension;
|
||||
import org.commonmark.node.FencedCodeBlock;
|
||||
import org.commonmark.parser.Parser;
|
||||
|
||||
import java.io.BufferedReader;
|
||||
import java.io.IOException;
|
||||
import java.io.InputStream;
|
||||
import java.io.InputStreamReader;
|
||||
import java.util.Collections;
|
||||
|
||||
import ru.noties.debug.AndroidLogDebugOutput;
|
||||
import ru.noties.debug.Debug;
|
||||
import ru.noties.markwon.AbstractMarkwonPlugin;
|
||||
import ru.noties.markwon.Markwon;
|
||||
import ru.noties.markwon.MarkwonConfiguration;
|
||||
import ru.noties.markwon.MarkwonVisitor;
|
||||
import ru.noties.markwon.core.CorePlugin;
|
||||
import ru.noties.markwon.ext.tables.TablePlugin;
|
||||
import ru.noties.markwon.html.HtmlPlugin;
|
||||
import ru.noties.markwon.image.ImagesPlugin;
|
||||
import ru.noties.markwon.image.svg.SvgPlugin;
|
||||
import ru.noties.markwon.recycler.MarkwonAdapter;
|
||||
import ru.noties.markwon.recycler.SimpleEntry;
|
||||
import ru.noties.markwon.sample.R;
|
||||
import ru.noties.markwon.syntax.SyntaxHighlightPlugin;
|
||||
import ru.noties.markwon.urlprocessor.UrlProcessor;
|
||||
import ru.noties.markwon.urlprocessor.UrlProcessorRelativeToAbsolute;
|
||||
|
||||
@ -54,7 +56,7 @@ public class RecyclerActivity extends Activity {
|
||||
// with `@+id/text` id
|
||||
.include(FencedCodeBlock.class, new SimpleEntry(R.layout.adapter_fenced_code_block))
|
||||
// create own implementation of entry for different rendering
|
||||
.include(TableBlock.class, new TableEntry())
|
||||
.include(TableBlock.class, new TableEntry2())
|
||||
// specify default entry (for all other blocks)
|
||||
.defaultEntry(new SimpleEntry(R.layout.adapter_default_entry))
|
||||
.build();
|
||||
@ -81,15 +83,35 @@ public class RecyclerActivity extends Activity {
|
||||
.usePlugin(CorePlugin.create())
|
||||
.usePlugin(ImagesPlugin.createWithAssets(context))
|
||||
.usePlugin(SvgPlugin.create(context.getResources()))
|
||||
// although we will be rendering table differently we still need
|
||||
// to register commonmark-java tables extension (which TablePlugin does)
|
||||
.usePlugin(TablePlugin.create(context))
|
||||
.usePlugin(new AbstractMarkwonPlugin() {
|
||||
@Override
|
||||
public void configureParser(@NonNull Parser.Builder builder) {
|
||||
// it's important NOT to use TablePlugin
|
||||
// the only thing we want from it is commonmark-java parser extension
|
||||
builder.extensions(Collections.singleton(TablesExtension.create()));
|
||||
}
|
||||
})
|
||||
.usePlugin(HtmlPlugin.create())
|
||||
// .usePlugin(SyntaxHighlightPlugin.create())
|
||||
.usePlugin(new AbstractMarkwonPlugin() {
|
||||
@Override
|
||||
public void configureConfiguration(@NonNull MarkwonConfiguration.Builder builder) {
|
||||
builder.urlProcessor(new UrlProcessorInitialReadme());
|
||||
}
|
||||
|
||||
@Override
|
||||
public void configureVisitor(@NonNull MarkwonVisitor.Builder builder) {
|
||||
builder.on(FencedCodeBlock.class, (visitor, fencedCodeBlock) -> {
|
||||
// we actually won't be applying code spans here, as our custom view will
|
||||
// draw background and apply mono typeface
|
||||
//
|
||||
// NB the `trim` operation on literal (as code will have a new line at the end)
|
||||
final CharSequence code = visitor.configuration()
|
||||
.syntaxHighlight()
|
||||
.highlight(fencedCodeBlock.getInfo(), fencedCodeBlock.getLiteral().trim());
|
||||
visitor.builder().append(code);
|
||||
});
|
||||
}
|
||||
})
|
||||
.build();
|
||||
}
|
||||
|
@ -17,7 +17,7 @@ import ru.noties.markwon.recycler.MarkwonAdapter;
|
||||
import ru.noties.markwon.sample.R;
|
||||
|
||||
// do not use in real applications, this is just a showcase
|
||||
public class TableEntry implements MarkwonAdapter.Entry<TableEntry.TableNodeHolder, TableBlock> {
|
||||
public class TableEntry implements MarkwonAdapter.Entry<TableBlock, TableEntry.TableNodeHolder> {
|
||||
|
||||
private final Map<TableBlock, Table> cache = new HashMap<>(2);
|
||||
|
||||
|
@ -0,0 +1,121 @@
|
||||
package ru.noties.markwon.sample.recycler;
|
||||
|
||||
import android.annotation.SuppressLint;
|
||||
import android.content.Context;
|
||||
import android.support.annotation.NonNull;
|
||||
import android.view.Gravity;
|
||||
import android.view.LayoutInflater;
|
||||
import android.view.View;
|
||||
import android.view.ViewGroup;
|
||||
import android.widget.TableLayout;
|
||||
import android.widget.TableRow;
|
||||
import android.widget.TextView;
|
||||
|
||||
import org.commonmark.ext.gfm.tables.TableBlock;
|
||||
|
||||
import java.util.HashMap;
|
||||
import java.util.Map;
|
||||
|
||||
import ru.noties.markwon.Markwon;
|
||||
import ru.noties.markwon.ext.tables.Table;
|
||||
import ru.noties.markwon.recycler.MarkwonAdapter;
|
||||
import ru.noties.markwon.sample.R;
|
||||
|
||||
public class TableEntry2 implements MarkwonAdapter.Entry<TableBlock, TableEntry2.TableHolder> {
|
||||
|
||||
private final Map<TableBlock, Table> map = new HashMap<>(3);
|
||||
|
||||
@NonNull
|
||||
@Override
|
||||
public TableHolder createHolder(@NonNull LayoutInflater inflater, @NonNull ViewGroup parent) {
|
||||
return new TableHolder(inflater.inflate(R.layout.adapter_table_block_2, parent, false));
|
||||
}
|
||||
|
||||
@Override
|
||||
public void bindHolder(@NonNull Markwon markwon, @NonNull TableHolder holder, @NonNull TableBlock node) {
|
||||
|
||||
Table table = map.get(node);
|
||||
if (table == null) {
|
||||
table = Table.parse(markwon, node);
|
||||
map.put(node, table);
|
||||
}
|
||||
|
||||
// check if this exact TableBlock was already
|
||||
final TableLayout layout = holder.layout;
|
||||
if (table == null
|
||||
|| table == layout.getTag(R.id.table_layout)) {
|
||||
return;
|
||||
}
|
||||
|
||||
layout.setTag(R.id.table_layout, table);
|
||||
layout.removeAllViews();
|
||||
layout.setBackgroundResource(R.drawable.bg_table_cell);
|
||||
|
||||
final Context context = layout.getContext();
|
||||
final LayoutInflater inflater = LayoutInflater.from(context);
|
||||
|
||||
TableRow tableRow;
|
||||
TextView textView;
|
||||
|
||||
for (Table.Row row : table.rows()) {
|
||||
tableRow = new TableRow(context);
|
||||
for (Table.Column column : row.columns()) {
|
||||
textView = (TextView) inflater.inflate(R.layout.view_table_entry_cell, tableRow, false);
|
||||
textView.setGravity(textGravity(column.alignment()));
|
||||
markwon.setParsedMarkdown(textView, column.content());
|
||||
textView.getPaint().setFakeBoldText(row.header());
|
||||
textView.setBackgroundResource(R.drawable.bg_table_cell);
|
||||
tableRow.addView(textView);
|
||||
}
|
||||
layout.addView(tableRow);
|
||||
}
|
||||
}
|
||||
|
||||
@Override
|
||||
public long id(@NonNull TableBlock node) {
|
||||
return node.hashCode();
|
||||
}
|
||||
|
||||
@Override
|
||||
public void clear() {
|
||||
map.clear();
|
||||
}
|
||||
|
||||
static class TableHolder extends MarkwonAdapter.Holder {
|
||||
|
||||
final TableLayout layout;
|
||||
|
||||
TableHolder(@NonNull View itemView) {
|
||||
super(itemView);
|
||||
|
||||
this.layout = requireView(R.id.table_layout);
|
||||
}
|
||||
}
|
||||
|
||||
// we will use gravity instead of textAlignment because min sdk is 16 (textAlignment starts at 17)
|
||||
@SuppressLint("RtlHardcoded")
|
||||
private static int textGravity(@NonNull Table.Alignment alignment) {
|
||||
|
||||
final int gravity;
|
||||
|
||||
switch (alignment) {
|
||||
|
||||
case LEFT:
|
||||
gravity = Gravity.LEFT;
|
||||
break;
|
||||
|
||||
case CENTER:
|
||||
gravity = Gravity.CENTER_HORIZONTAL;
|
||||
break;
|
||||
|
||||
case RIGHT:
|
||||
gravity = Gravity.RIGHT;
|
||||
break;
|
||||
|
||||
default:
|
||||
throw new IllegalStateException("Unknown table alignment: " + alignment);
|
||||
}
|
||||
|
||||
return gravity | Gravity.CENTER_VERTICAL;
|
||||
}
|
||||
}
|
@ -54,7 +54,6 @@ public class TableEntryView extends LinearLayout {
|
||||
|
||||
rowEvenBackgroundColor = array.getColor(R.styleable.TableEntryView_tev_rowEvenBackgroundColor, 0);
|
||||
|
||||
|
||||
final int stroke = array.getDimensionPixelSize(R.styleable.TableEntryView_tev_borderWidth, 0);
|
||||
|
||||
// half of requested
|
||||
@ -124,6 +123,8 @@ public class TableEntryView extends LinearLayout {
|
||||
textView.setText(column.content());
|
||||
textView.getPaint().setFakeBoldText(row.header());
|
||||
}
|
||||
|
||||
group.requestLayout();
|
||||
}
|
||||
|
||||
@NonNull
|
||||
|
@ -1,5 +1,6 @@
|
||||
<?xml version="1.0" encoding="utf-8"?>
|
||||
<TextView xmlns:android="http://schemas.android.com/apk/res/android"
|
||||
xmlns:tools="http://schemas.android.com/tools"
|
||||
android:id="@+id/text"
|
||||
android:layout_width="match_parent"
|
||||
android:layout_height="wrap_content"
|
||||
@ -10,4 +11,5 @@
|
||||
android:paddingBottom="8dip"
|
||||
android:textAppearance="?android:attr/textAppearanceMedium"
|
||||
android:textColor="#000"
|
||||
android:textSize="16sp" />
|
||||
android:textSize="16sp"
|
||||
tools:text="Hello" />
|
@ -14,11 +14,15 @@
|
||||
android:id="@+id/text"
|
||||
android:layout_width="wrap_content"
|
||||
android:layout_height="wrap_content"
|
||||
android:background="#0f000000"
|
||||
android:fontFamily="monospace"
|
||||
android:lineSpacingExtra="2dip"
|
||||
android:paddingLeft="16dip"
|
||||
android:paddingTop="8dip"
|
||||
android:paddingRight="16dip"
|
||||
android:paddingBottom="8dip"
|
||||
android:textAppearance="?android:attr/textAppearanceMedium"
|
||||
android:textSize="16sp"
|
||||
tools:text="# Hello there! and taskjs" />
|
||||
android:textSize="14sp"
|
||||
tools:text="# Hello there! and tasks" />
|
||||
|
||||
</HorizontalScrollView>
|
@ -5,7 +5,6 @@
|
||||
android:layout_height="wrap_content"
|
||||
android:clipChildren="false"
|
||||
android:clipToPadding="false"
|
||||
android:fillViewport="true"
|
||||
android:paddingLeft="16dip"
|
||||
android:paddingTop="8dip"
|
||||
android:paddingRight="16dip"
|
||||
|
@ -0,0 +1,18 @@
|
||||
<?xml version="1.0" encoding="utf-8"?>
|
||||
<HorizontalScrollView xmlns:android="http://schemas.android.com/apk/res/android"
|
||||
android:layout_width="match_parent"
|
||||
android:layout_height="wrap_content"
|
||||
android:clipChildren="false"
|
||||
android:clipToPadding="false"
|
||||
android:paddingLeft="16dip"
|
||||
android:paddingTop="8dip"
|
||||
android:paddingRight="16dip"
|
||||
android:paddingBottom="8dip"
|
||||
android:scrollbarStyle="outsideInset">
|
||||
|
||||
<TableLayout
|
||||
android:id="@+id/table_layout"
|
||||
android:layout_width="wrap_content"
|
||||
android:layout_height="wrap_content" />
|
||||
|
||||
</HorizontalScrollView>
|
@ -1,10 +1,10 @@
|
||||
<?xml version="1.0" encoding="utf-8"?>
|
||||
<TextView xmlns:android="http://schemas.android.com/apk/res/android"
|
||||
xmlns:tools="http://schemas.android.com/tools"
|
||||
android:layout_width="0px"
|
||||
android:layout_width="wrap_content"
|
||||
android:layout_height="match_parent"
|
||||
android:layout_weight="1"
|
||||
android:padding="4dip"
|
||||
android:gravity="center_vertical"
|
||||
android:padding="8dip"
|
||||
android:textAppearance="?android:attr/textAppearanceMedium"
|
||||
android:textColor="#000"
|
||||
android:textSize="16sp"
|
||||
|
@ -1,5 +1,5 @@
|
||||
<?xml version="1.0" encoding="utf-8"?>
|
||||
<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
|
||||
android:layout_width="match_parent"
|
||||
android:layout_width="wrap_content"
|
||||
android:layout_height="wrap_content"
|
||||
android:orientation="horizontal" />
|
6
sample/src/main/res/drawable/bg_table_cell.xml
Normal file
6
sample/src/main/res/drawable/bg_table_cell.xml
Normal file
@ -0,0 +1,6 @@
|
||||
<?xml version="1.0" encoding="utf-8"?>
|
||||
<shape xmlns:android="http://schemas.android.com/apk/res/android">
|
||||
<stroke
|
||||
android:width="0.5dip"
|
||||
android:color="#000" />
|
||||
</shape>
|
Loading…
x
Reference in New Issue
Block a user