diff --git a/android/AndroidManifest.xml b/android/AndroidManifest.xml
index 940421fc08..f80a2e0403 100644
--- a/android/AndroidManifest.xml
+++ b/android/AndroidManifest.xml
@@ -443,6 +443,9 @@
android:configChanges="keyboardHidden|orientation|screenSize">
+
+
+
+
+
diff --git a/android/res/layout/tag_item.xml b/android/res/layout/tag_item.xml
new file mode 100644
index 0000000000..570319842d
--- /dev/null
+++ b/android/res/layout/tag_item.xml
@@ -0,0 +1,21 @@
+
+
+
+
+
+
diff --git a/android/res/layout/tags_recycler.xml b/android/res/layout/tags_recycler.xml
new file mode 100644
index 0000000000..9cfb2d2472
--- /dev/null
+++ b/android/res/layout/tags_recycler.xml
@@ -0,0 +1,11 @@
+
+
diff --git a/android/res/layout/ugc_routes_frag.xml b/android/res/layout/ugc_routes_frag.xml
new file mode 100644
index 0000000000..99e70ea2be
--- /dev/null
+++ b/android/res/layout/ugc_routes_frag.xml
@@ -0,0 +1,71 @@
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
diff --git a/android/res/menu/menu_done.xml b/android/res/menu/menu_done.xml
new file mode 100644
index 0000000000..d8c065e6f8
--- /dev/null
+++ b/android/res/menu/menu_done.xml
@@ -0,0 +1,10 @@
+
+
diff --git a/android/res/values/strings.xml b/android/res/values/strings.xml
index df38b08387..cd32dcb626 100644
--- a/android/res/values/strings.xml
+++ b/android/res/values/strings.xml
@@ -2402,4 +2402,7 @@
wheelchair-limited
wheelchair-no
wheelchair-yes
+ Please, select tags to help other travelers find your guide. This is mandatory.
+ Loading tagsā¦
+ Select tags
diff --git a/android/src/com/mapswithme/maps/bookmarks/CachedBookmarkCategoriesFragment.java b/android/src/com/mapswithme/maps/bookmarks/CachedBookmarkCategoriesFragment.java
index dd7569a0d0..3618bb69ff 100644
--- a/android/src/com/mapswithme/maps/bookmarks/CachedBookmarkCategoriesFragment.java
+++ b/android/src/com/mapswithme/maps/bookmarks/CachedBookmarkCategoriesFragment.java
@@ -16,6 +16,7 @@ import com.mapswithme.maps.bookmarks.data.BookmarkCategory;
import com.mapswithme.maps.bookmarks.data.BookmarkManager;
import com.mapswithme.maps.bookmarks.data.CatalogCustomProperty;
import com.mapswithme.maps.bookmarks.data.CatalogTagsGroup;
+import com.mapswithme.maps.ugc.routes.UgcRouteTagsActivity;
import com.mapswithme.util.SharedPropertiesUtils;
import com.mapswithme.util.UiUtils;
import com.mapswithme.util.sharing.TargetUtils;
@@ -131,8 +132,8 @@ public class CachedBookmarkCategoriesFragment extends BaseBookmarkCategoriesFrag
private void openBookmarksCatalogScreen()
{
- BookmarksCatalogActivity.startForResult(this, BookmarksCatalogActivity.REQ_CODE_CATALOG);
- Statistics.INSTANCE.trackOpenCatalogScreen();
+ Intent intent = new Intent(getContext(), UgcRouteTagsActivity.class);
+ startActivity(intent);
}
@Override
diff --git a/android/src/com/mapswithme/maps/bookmarks/data/BookmarkManager.java b/android/src/com/mapswithme/maps/bookmarks/data/BookmarkManager.java
index f036784b82..93f276f93c 100644
--- a/android/src/com/mapswithme/maps/bookmarks/data/BookmarkManager.java
+++ b/android/src/com/mapswithme/maps/bookmarks/data/BookmarkManager.java
@@ -600,6 +600,11 @@ public enum BookmarkManager
return nativeGetCatalogFrontendUrl();
}
+ public void requestRouteTags()
+ {
+ nativeRequestCatalogTags();
+ }
+
public boolean isCategoryFromCatalog(long catId)
{
return nativeIsCategoryFromCatalog(catId);
diff --git a/android/src/com/mapswithme/maps/bookmarks/data/CatalogTag.java b/android/src/com/mapswithme/maps/bookmarks/data/CatalogTag.java
index fe0bc08bd2..8fa1d2e85c 100644
--- a/android/src/com/mapswithme/maps/bookmarks/data/CatalogTag.java
+++ b/android/src/com/mapswithme/maps/bookmarks/data/CatalogTag.java
@@ -1,9 +1,11 @@
package com.mapswithme.maps.bookmarks.data;
import android.graphics.Color;
+import android.os.Parcel;
+import android.os.Parcelable;
import android.support.annotation.NonNull;
-public class CatalogTag
+public class CatalogTag implements Parcelable
{
@NonNull
private final String mId;
@@ -27,4 +29,55 @@ public class CatalogTag
public String getLocalizedName() { return mLocalizedName; }
public int getColor() { return mColor; }
+
+ @Override
+ public int describeContents()
+ {
+ return 0;
+ }
+
+ @Override
+ public void writeToParcel(Parcel dest, int flags)
+ {
+ dest.writeString(this.mId);
+ dest.writeString(this.mLocalizedName);
+ dest.writeInt(this.mColor);
+ }
+
+ protected CatalogTag(Parcel in)
+ {
+ this.mId = in.readString();
+ this.mLocalizedName = in.readString();
+ this.mColor = in.readInt();
+ }
+
+ public static final Creator CREATOR = new Creator()
+ {
+ @Override
+ public CatalogTag createFromParcel(Parcel source)
+ {
+ return new CatalogTag(source);
+ }
+
+ @Override
+ public CatalogTag[] newArray(int size)
+ {
+ return new CatalogTag[size];
+ }
+ };
+
+ @Override
+ public boolean equals(Object o)
+ {
+ if (this == o) return true;
+ if (o == null || getClass() != o.getClass()) return false;
+ CatalogTag that = (CatalogTag) o;
+ return mId.equals(that.mId) || mId.equals(that.mId);
+ }
+
+ @Override
+ public int hashCode()
+ {
+ return mId.hashCode();
+ }
}
diff --git a/android/src/com/mapswithme/maps/bookmarks/data/CatalogTagsGroup.java b/android/src/com/mapswithme/maps/bookmarks/data/CatalogTagsGroup.java
index d77bafc36d..26305b9caa 100644
--- a/android/src/com/mapswithme/maps/bookmarks/data/CatalogTagsGroup.java
+++ b/android/src/com/mapswithme/maps/bookmarks/data/CatalogTagsGroup.java
@@ -2,23 +2,28 @@ package com.mapswithme.maps.bookmarks.data;
import android.support.annotation.NonNull;
+import java.util.Arrays;
+import java.util.Collection;
+import java.util.Collections;
+import java.util.List;
+
public class CatalogTagsGroup
{
@NonNull
private final String mLocalizedName;
@NonNull
- private final CatalogTag[] mTags;
+ private final List mTags;
public CatalogTagsGroup(@NonNull String localizedName, @NonNull CatalogTag[] tags)
{
mLocalizedName = localizedName;
- mTags = tags;
+ mTags = Collections.unmodifiableList(Arrays.asList(tags));
}
@NonNull
public String getLocalizedName() { return mLocalizedName; }
@NonNull
- public CatalogTag[] getTags() { return mTags; }
+ public List getTags() { return mTags; }
}
diff --git a/android/src/com/mapswithme/maps/ugc/routes/TagsResFactory.java b/android/src/com/mapswithme/maps/ugc/routes/TagsResFactory.java
new file mode 100644
index 0000000000..655a9d274e
--- /dev/null
+++ b/android/src/com/mapswithme/maps/ugc/routes/TagsResFactory.java
@@ -0,0 +1,59 @@
+package com.mapswithme.maps.ugc.routes;
+
+import android.content.Context;
+import android.content.res.ColorStateList;
+import android.content.res.Resources;
+import android.graphics.Color;
+import android.graphics.drawable.ColorDrawable;
+import android.graphics.drawable.Drawable;
+import android.graphics.drawable.GradientDrawable;
+import android.graphics.drawable.LayerDrawable;
+import android.graphics.drawable.ShapeDrawable;
+import android.graphics.drawable.StateListDrawable;
+import android.graphics.drawable.shapes.RectShape;
+import android.support.annotation.NonNull;
+
+import com.mapswithme.maps.R;
+
+public class TagsResFactory
+{
+ public static StateListDrawable makeSelector(@NonNull Context context, int color)
+ {
+ StateListDrawable drawable = new StateListDrawable();
+ drawable.addState(new int[] { android.R.attr.state_selected }, getSelectedDrawable(color));
+ drawable.addState(new int[] {}, getDefaultDrawable(context, color));
+ return drawable;
+ }
+
+ private static Drawable getDefaultDrawable(@NonNull Context context, int color)
+ {
+ Resources res = context.getResources();
+ GradientDrawable gradientDrawable = new GradientDrawable();
+ gradientDrawable.setStroke(res.getDimensionPixelSize(R.dimen.divider_height), color);
+
+ ShapeDrawable shapeDrawable = new ShapeDrawable(new RectShape());
+ shapeDrawable.getPaint().setColor(Color.WHITE);
+ return new LayerDrawable(new Drawable[] { shapeDrawable, gradientDrawable });
+ }
+
+ @NonNull
+ private static ColorDrawable getSelectedDrawable(int color)
+ {
+ return new ColorDrawable(color);
+ }
+
+ @NonNull
+ public static ColorStateList makeColor(@NonNull Context context, int color)
+ {
+ return new ColorStateList(
+ new int[][] {
+ new int[] { android.R.attr.state_selected },
+ new int[] {}
+ },
+ new int[] {
+ context.getResources().getColor(android.R.color.white),
+ color
+ }
+ );
+ }
+}
diff --git a/android/src/com/mapswithme/maps/ugc/routes/UgcRouteTagsActivity.java b/android/src/com/mapswithme/maps/ugc/routes/UgcRouteTagsActivity.java
new file mode 100644
index 0000000000..4272d4ccbc
--- /dev/null
+++ b/android/src/com/mapswithme/maps/ugc/routes/UgcRouteTagsActivity.java
@@ -0,0 +1,16 @@
+package com.mapswithme.maps.ugc.routes;
+
+import android.support.v4.app.Fragment;
+
+import com.mapswithme.maps.base.BaseToolbarActivity;
+
+public class UgcRouteTagsActivity extends BaseToolbarActivity
+{
+ public static final String EXTRA_TAGS = "selected_tags";
+
+ @Override
+ protected Class extends Fragment> getFragmentClass()
+ {
+ return UgcRoutesFragment.class;
+ }
+}
diff --git a/android/src/com/mapswithme/maps/ugc/routes/UgcRoutesFragment.java b/android/src/com/mapswithme/maps/ugc/routes/UgcRoutesFragment.java
new file mode 100644
index 0000000000..01e1d0fc78
--- /dev/null
+++ b/android/src/com/mapswithme/maps/ugc/routes/UgcRoutesFragment.java
@@ -0,0 +1,317 @@
+package com.mapswithme.maps.ugc.routes;
+
+import android.app.Activity;
+import android.content.Intent;
+import android.content.res.ColorStateList;
+import android.content.res.Resources;
+import android.graphics.drawable.Drawable;
+import android.graphics.drawable.StateListDrawable;
+import android.os.Bundle;
+import android.support.annotation.NonNull;
+import android.support.annotation.Nullable;
+import android.support.v7.widget.RecyclerView;
+import android.view.LayoutInflater;
+import android.view.Menu;
+import android.view.MenuInflater;
+import android.view.MenuItem;
+import android.view.View;
+import android.view.ViewGroup;
+import android.widget.TextView;
+
+import com.google.android.flexbox.FlexboxItemDecoration;
+import com.google.android.flexbox.FlexboxLayoutManager;
+import com.mapswithme.maps.R;
+import com.mapswithme.maps.base.BaseMwmFragment;
+import com.mapswithme.maps.bookmarks.OnItemClickListener;
+import com.mapswithme.maps.bookmarks.data.BookmarkManager;
+import com.mapswithme.maps.bookmarks.data.CatalogTag;
+import com.mapswithme.maps.bookmarks.data.CatalogTagsGroup;
+import com.mapswithme.maps.widget.recycler.TagItemDecoration;
+import com.mapswithme.maps.widget.recycler.UgcRouteTagItemDecorator;
+import com.mapswithme.util.UiUtils;
+
+import java.util.ArrayList;
+import java.util.Collection;
+import java.util.Collections;
+import java.util.List;
+import java.util.Objects;
+
+public class UgcRoutesFragment extends BaseMwmFragment implements BookmarkManager.BookmarksCatalogListener
+
+{
+ @SuppressWarnings("NullableProblems")
+ @NonNull
+ private List mRecycler = new ArrayList<>();
+
+ @SuppressWarnings("NullableProblems")
+ @NonNull
+ private View mProgress;
+
+ @SuppressWarnings("NullableProblems")
+ @NonNull
+ private View mRetryBtnContainer;
+
+ @SuppressWarnings("NullableProblems")
+ @NonNull
+ private List mAdapter = new ArrayList<>();
+
+ @SuppressWarnings("NullableProblems")
+ @NonNull
+ private ViewGroup mTagsContainer;
+
+ @Nullable
+ @Override
+ public View onCreateView(LayoutInflater inflater, @Nullable ViewGroup container, @Nullable Bundle savedInstanceState)
+ {
+ LayoutInflater layoutInflater = LayoutInflater.from(getContext());
+ ViewGroup root = (ViewGroup) layoutInflater.inflate(R.layout.ugc_routes_frag, container,
+ false);
+ setHasOptionsMenu(true);
+ mProgress = root.findViewById(R.id.progress_container);
+ mTagsContainer = root.findViewById(R.id.tags_container);
+ mRetryBtnContainer = root.findViewById(R.id.retry_btn_container);
+ View retryBtn = mRetryBtnContainer.findViewById(R.id.retry_btn);
+ retryBtn.setOnClickListener(v -> onRetryClicked());
+ UiUtils.hide(mTagsContainer, mRetryBtnContainer);
+ UiUtils.show(mProgress);
+ BookmarkManager.INSTANCE.requestRouteTags();
+ return root;
+ }
+
+ private void onRetryClicked()
+ {
+ UiUtils.hide(mTagsContainer, mRetryBtnContainer);
+ UiUtils.show(mProgress);
+ BookmarkManager.INSTANCE.requestRouteTags();
+ }
+
+ @Override
+ public void onCreateOptionsMenu(Menu menu, MenuInflater inflater)
+ {
+ inflater.inflate(R.menu.menu_done, menu);
+ }
+
+ @Override
+ public boolean onOptionsItemSelected(MenuItem item)
+ {
+ if (item.getItemId() == R.id.done)
+ {
+ ArrayList catalogTags = new ArrayList();
+ for (TagsAdapter each : mAdapter)
+ {
+ catalogTags.addAll(each.getSelectedTags());
+ }
+ Intent result = new Intent().putParcelableArrayListExtra(UgcRouteTagsActivity.EXTRA_TAGS, catalogTags);
+ getActivity().setResult(Activity.RESULT_OK, result);
+ getActivity().finish();
+ return true;
+ }
+
+ return super.onOptionsItemSelected(item);
+ }
+
+ @Override
+ public void onStart()
+ {
+ super.onStart();
+ BookmarkManager.INSTANCE.addCatalogListener(this);
+ }
+
+ @Override
+ public void onStop()
+ {
+ super.onStop();
+ BookmarkManager.INSTANCE.removeCatalogListener(this);
+ }
+
+ @Override
+ public void onImportStarted(@NonNull String serverId)
+ {
+
+ }
+
+ @Override
+ public void onImportFinished(@NonNull String serverId, long catId, boolean successful)
+ {
+
+ }
+
+ @Override
+ public void onTagsReceived(boolean successful, @NonNull CatalogTagsGroup[] tagsGroups)
+ {
+ UiUtils.showIf(successful && tagsGroups.length != 0, mTagsContainer);
+ UiUtils.hideIf(successful && tagsGroups.length != 0, mRetryBtnContainer);
+ UiUtils.hide(mProgress);
+
+ if (tagsGroups.length == 0)
+ return;
+ addTags(tagsGroups);
+ }
+
+ private void addTags(@NonNull CatalogTagsGroup[] tagsGroups)
+ {
+ for (CatalogTagsGroup tagsGroup : tagsGroups)
+ {
+ RecyclerView recycler = (RecyclerView) getLayoutInflater().inflate(R.layout.tags_recycler,
+ mTagsContainer,
+ false);
+
+ mTagsContainer.addView(recycler);
+ FlexboxLayoutManager layoutManager = new FlexboxLayoutManager(getContext());
+ Resources res = getResources();
+ Drawable divider = res.getDrawable(R.drawable.flexbox_divider);
+ TagItemDecoration decor = new UgcRouteTagItemDecorator(divider);
+ recycler.addItemDecoration(decor);
+ recycler.setLayoutManager(layoutManager);
+ final TagsAdapter adapter = new TagsAdapter((v, item) -> {});
+ recycler.setAdapter(adapter);
+ recycler.setItemAnimator(null);
+ recycler.setNestedScrollingEnabled(false);
+ adapter.setTags(tagsGroup.getTags());
+ mAdapter.add(adapter);
+ mRecycler.add(recycler);
+ }
+ }
+
+ @Override
+ public void onUploadStarted(long originCategoryId)
+ {
+
+ }
+
+ @Override
+ public void onUploadFinished(int uploadResult, @NonNull String description,
+ long originCategoryId, long resultCategoryId)
+ {
+
+ }
+
+ private static class TagsAdapter extends RecyclerView.Adapter
+ {
+ @NonNull
+ private final SelectionState mState;
+
+ @NonNull
+ private final OnItemClickListener mListener;
+
+ @NonNull
+ private List mTags = Collections.emptyList();
+
+ TagsAdapter(@NonNull OnItemClickListener listener)
+ {
+ mListener = new ClickListenerWrapper(listener);
+ mState = new SelectionState();
+ setHasStableIds(true);
+ }
+
+ @Override
+ public TagViewHolder onCreateViewHolder(ViewGroup parent, int viewType)
+ {
+ View itemView = LayoutInflater.from(parent.getContext()).inflate(R.layout.tag_item, parent, false);
+ return new TagViewHolder(itemView, mListener);
+ }
+
+ @Override
+ public long getItemId(int position)
+ {
+ return mTags.get(position).hashCode();
+ }
+
+ @Override
+ public void onBindViewHolder(TagViewHolder holder, int position)
+ {
+ CatalogTag tag = mTags.get(position);
+ holder.itemView.setSelected(mState.contains(tag));
+ holder.mTag = tag;
+ StateListDrawable selector = TagsResFactory.makeSelector(holder.itemView.getContext(),tag.getColor());
+ holder.itemView.setBackgroundDrawable(selector);
+ ColorStateList color = TagsResFactory.makeColor(holder.mText.getContext(), tag.getColor());
+ holder.mText.setTextColor(color);
+ holder.mText.setText(tag.getLocalizedName());
+ }
+
+ @Override
+ public int getItemCount()
+ {
+ return mTags.size();
+ }
+
+ public void setTags(@NonNull List tags)
+ {
+ mTags = tags;
+ notifyDataSetChanged();
+ }
+
+ Collection getSelectedTags()
+ {
+ return Collections.unmodifiableCollection(mTags);
+ }
+
+ private static final class SelectionState {
+ @NonNull
+ private final List mTags = new ArrayList<>();
+
+ private void add(@NonNull CatalogTag tag){
+ mTags.add(tag);
+ }
+
+ private boolean remove(@NonNull CatalogTag tag)
+ {
+ return mTags.remove(tag);
+ }
+
+ private boolean contains(@NonNull CatalogTag tag)
+ {
+ return mTags.contains(tag);
+ }
+ }
+
+ private class ClickListenerWrapper implements OnItemClickListener
+ {
+ @NonNull
+ private final OnItemClickListener mListener;
+
+ public ClickListenerWrapper(@NonNull OnItemClickListener listener)
+ {
+ mListener = listener;
+ }
+
+ @Override
+ public void onItemClick(@NonNull View v, @NonNull TagViewHolder item)
+ {
+ if (mState.contains(item.getEntity()))
+ mState.remove(item.getEntity());
+ else
+ mState.add(item.getEntity());
+
+ mListener.onItemClick(v, item);
+
+ notifyItemChanged(item.getAdapterPosition());
+ }
+ }
+ }
+
+ static class TagViewHolder extends RecyclerView.ViewHolder
+ {
+ @NonNull
+ private final TextView mText;
+ @NonNull
+ private final OnItemClickListener mListener;
+ @Nullable
+ private CatalogTag mTag;
+
+ TagViewHolder(View itemView, @NonNull OnItemClickListener listener)
+ {
+ super(itemView);
+ mText = itemView.findViewById(R.id.text);
+ mListener = listener;
+ itemView.setOnClickListener(v -> mListener.onItemClick(v, this));
+ }
+
+ @NonNull
+ private CatalogTag getEntity()
+ {
+ return Objects.requireNonNull(mTag);
+ }
+ }
+}
diff --git a/android/src/com/mapswithme/maps/widget/recycler/TagItemDecoration.java b/android/src/com/mapswithme/maps/widget/recycler/TagItemDecoration.java
index bfd7085238..ddcbe254eb 100644
--- a/android/src/com/mapswithme/maps/widget/recycler/TagItemDecoration.java
+++ b/android/src/com/mapswithme/maps/widget/recycler/TagItemDecoration.java
@@ -80,4 +80,10 @@ public class TagItemDecoration extends RecyclerView.ItemDecoration
outRect.left = mDivider.getIntrinsicWidth();
outRect.top = mDivider.getIntrinsicHeight();
}
+
+ @NonNull
+ protected Drawable getDivider()
+ {
+ return mDivider;
+ }
}
diff --git a/android/src/com/mapswithme/maps/widget/recycler/UgcRouteTagItemDecorator.java b/android/src/com/mapswithme/maps/widget/recycler/UgcRouteTagItemDecorator.java
new file mode 100644
index 0000000000..cf32562794
--- /dev/null
+++ b/android/src/com/mapswithme/maps/widget/recycler/UgcRouteTagItemDecorator.java
@@ -0,0 +1,48 @@
+package com.mapswithme.maps.widget.recycler;
+
+import android.graphics.Rect;
+import android.graphics.drawable.Drawable;
+import android.support.annotation.NonNull;
+import android.support.v7.widget.RecyclerView;
+import android.view.View;
+
+public class UgcRouteTagItemDecorator extends TagItemDecoration
+{
+ public UgcRouteTagItemDecorator(@NonNull Drawable divider)
+ {
+ super(divider);
+ }
+
+ private int mCurrentOffset;
+
+ @Override
+ public void getItemOffsets(Rect outRect, View view, RecyclerView parent, RecyclerView.State state)
+ {
+ super.getItemOffsets(outRect, view, parent, state);
+
+ if (isLineFull(outRect, view, parent))
+ {
+ mCurrentOffset = 0;
+ }
+ else
+ {
+ mCurrentOffset += view.getWidth() + outRect.left;
+ }
+
+ if (mCurrentOffset == 0)
+ {
+ outRect.left = 0;
+ outRect.right = getDivider().getIntrinsicWidth() / 2;
+ }
+ else
+ {
+ outRect.left = getDivider().getIntrinsicWidth() / 2;
+ outRect.right = getDivider().getIntrinsicWidth() / 2;
+ }
+ }
+
+ private boolean isLineFull(Rect outRect, View view, RecyclerView parent)
+ {
+ return mCurrentOffset + view.getWidth() + outRect.left > parent.getWidth() - (parent.getPaddingLeft() + parent.getRight());
+ }
+}
diff --git a/omim_config.h b/omim_config.h
index d6b3a95de0..f2992c64f1 100644
--- a/omim_config.h
+++ b/omim_config.h
@@ -1,9 +1,9 @@
#pragma once
-//#define TEST_ADS_REMOVAL_IDS
-//#define STAGE_BOOKMARKS_CATALOG_SERVER
-//#define STAGE_CLOUD_SERVER
-//#define STAGE_LOCALS_SERVER
-//#define STAGE_PASSPORT_SERVER
-//#define STAGE_PURCHASE_SERVER
-//#define STAGE_UGC_SERVER
+#define TEST_ADS_REMOVAL_IDS
+#define STAGE_BOOKMARKS_CATALOG_SERVER
+#define STAGE_CLOUD_SERVER
+#define STAGE_LOCALS_SERVER
+#define STAGE_PASSPORT_SERVER
+#define STAGE_PURCHASE_SERVER
+#define STAGE_UGC_SERVER