##Table of Contents
- Overview
- Setup
- 2.1 Run the Demo
- 2.2 Install the SDK
- 2.3 Add User Permissions
- Initialization
- Solution APIs
- 4.1 Visually Similar Recommendations
- 4.2 Search by Image
- 4.2.1 Selection Box
- 4.2.2 Resizing Settings
- 4.3 Search by Color
- Search Results
- Advanced Search Parameters
- Event Tracking
ViSearch is an API that provides accurate, reliable and scalable image search. ViSearch API provides two services ( Data API and Search API) to let the developers prepare image database and perform image searches efficiently. ViSearch API can be easily integrated into your web and mobile applications. For more details, see ViSearch API Documentation.
The ViSearch Androi SDK is an open source software to provide easy integration of ViSearch Search API with your Android mobile applications. It provides three search methods based on the ViSearch Search API - pre-indexed search, color search and upload search. For source code and references, please visit the Github Repository.
Current stable version: 1.3.3
Minimum Android SDK Version: API 9, Android 2.3
The source code of a demo application is provided together with the SDK (demo). You can simply open visearch-sdk-android project in Android Studio and run the cameraDemo project.
You should change the app key your own app key before running. You can create and mange
public class MainActivity extends FragmentActivity {
//TODO: init visearch app key here
private static final String appKey = "your_app_key ";
@Override
public void onCreate(Bundle savedInstanceState) {
...
You can play around with our demo app to see how we build up the cool image search feature using ViSearch SDK.
You can include the dependency in your project using gradle:
implementation 'com.visenze:visearch-android:1.3.3'
In the build.gradle
file under your app module, add the packaging options to ensure a successful compilation:
android {
...
packagingOptions {
exclude 'META-INF/NOTICE'
exclude 'META-INF/LICENSE'
}
...
}
If you want to use the packaged Jars directly in your project, please find all the dependencies in the directory /dependency
ViSearch Android SDK needs these user permissions to work. Add the following declarations to the AndroidManifest.xml
file. Network permission allows the app to connect to network services. Write/read to external storage permissions allow the app to load and save images on the device.
<?xml version="1.0" encoding="utf-8"?>
<manifest xmlns:android="http://schemas.android.com/apk/res/android"
package="com.visenze.android.visenze_demo_test">
<uses-permission android:name="android.permission.CAMERA"/>
<uses-permission android:name="android.permission.INTERNET"/>
<uses-permission android:name="android.permission.WRITE_EXTERNAL_STORAGE"/>
<application>
...
</application>
</manifest>
ViSearch
must be initialized with an app key before it can be used. In order for it to be notified of the search result, ViSearch.ResultListener
must be implemented. Call viSearch.setListener
to set the listener.
public class MyActivity extends Activity implements ViSearch.ResultListener{
//Please change to your app key
private static final String APP_KEY = "your_app_key";
...
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
ViSearch viSearch = new ViSearch.Builder(APP_KEY).build(context);
viSearch.setListener(this);
...
}
...
}
Please init ViSearch client in this way if you connect to another endpoint rather than default (https://visearch.visenze.com)
ViSearch viSearch = new ViSearch
.Builder(APP_KEY)
.setApiEndPoint("https://custom-visearch.yourdomain.com")
.build(context);
GET /search
Visually Similar Recommendations solution is to search for visually similar images in the image database giving an indexed image’s unique identifier (im_name).
IdSearchParams idSearchParams = new IdSearchParams("dress_1");
viSearch.idSearch(idSearchParams);
POST /uploadsearch
Search by image solution is to search similar images by uploading an image or providing an image url. Image class is used to perform the image encoding and resizing. You should construct the Image object and pass it to uploadsearch to start a search.
- Using an image from a local file path:
Image image = new Image("/local/path/to/image.jpg");
UploadSearchParams uploadSearchParams.setImage(image);
viSearch.uploadSearch(uploadSearchParams);
- Using an image by providing the Uri of the image in photo gallery:
Image image = new Image(context, uri);
UploadSearchParams uploadSearchParams = new UploadSearchParams(image);
viSearcher.uploadSearch(uploadSearchParams);
- Construct the
image
from the byte array returned by the camera preview callback:
@Override
public void onPictureTaken(byte[] bytes, Camera camera) {
Image image = new Image(bytes);
UploadSearchParams uploadSearchParams = new UploadSearchParams(image);
viSearcher.uploadSearch(uploadSearchParams);
}
- Alternatively, you can pass an image url directly to
uploadSearchParams
to start the search :
String url = "http://mydomain.com/sample_image.jpg";
UploadSearchParams uploadSearchParams = new UploadSearchParams(url);
viSearcher.uploadSearch(uploadSearchParams);
- If you are performing refinement on an uploaded image, you can pass the im_id returned in the search result to start the search instead of uploading the image again:
String imId;
@Override
public void onSearchResult(ResultList resultList) {
imId = resultList.getImId();
for (ImageResult imageResult : resultList.getImageList()) {
//Do something with the result
...
}
}
UploadSearchParams uploadSearchParams = new UploadSearchParams();
uploadSearchParams.setImId(imId);
viSearcher.uploadSearch(uploadSearchParams);
If the object you wish to search for takes up only a small portion of your image, or other irrelevant objects exists in the same image, chances are the search result could become inaccurate. Use the Box parameter to refine the search area of the image to improve accuracy. The box coordinated is set with respect to the original size of the uploading image:
Image image = new Image(this, uri);
// create the box to refine the area on the searching image
// Box(x1, y1, x2, y2) where (0,0) is the top-left corner
// of the image, (x1, y1) is the top-left corner of the box,
// and (x2, y2) is the bottom-right corner of the box.
image.setBox(0, 0, 400, 400);
If you are using im_url or im_id for upload search. You should pass the box in this way:
UploadSearchParams uploadSearchParams = new UploadSearchParams();
uploadSearchParams.setImId(imId);
uploadSearchParams.setBox(new Box(0, 0, 400, 400));
viSearcher.uploadSearch(uploadSearchParams);
When performing upload search, you may notice the increased search latency with increased image file size. This is due to the increased time spent in network transferring your images to the ViSearch server, and the increased time for processing larger image files in ViSearch.
To reduce upload search latency, by default the uploadSearch method makes a copy of your image file and resizes the copy to 512x512 pixels if both of the original dimensions exceed 512 pixels. This is the optimized size to lower search latency while not sacrificing search accuracy for general use cases:
- Image from local path or photo gallery
//default resize setting, set the image size to 512 x 512
Image image = new Image(imagePath, ResizeSettings.STANDARD);
If your image contains fine details such as textile patterns and textures, you can use an image with larger size for search to get better search result:
//for images with fine details, use HIGH resize settings 1024 x 1024
Image image = new Image(imagePath, ResizeSettings.HIGH);
Or, provide the customized resize settings. To make efficient use the of the memory and network bandwidth of mobile device, the maximum size is set at 1024 x 1024. Any image exceeds the limit will be resized to the limit:
//resize the image to 800 by 800 area using jpeg 80 quality
Image image = new Image(imagePath, new ResizeSettings(800, 800, 80));
- Image from camera callback
ViSearch Android SDK provides an interface to handle byte array returned from Camera.PictureCallback
. Use ResizeSettings.CAMERA_STANDARD
and ResizeSettings.CAMERA_HIGH
to configure the resize settings. The image taken from the camera might not be in the desired orientaiton, a rotation parameter can be set to rotate the image to the correct orientation:
@Override
public void onPictureTaken(byte[] bytes, Camera camera) {
Image image = new Image(bytes, ResizeSettings.CAMERA_HIGH, 90);
UploadSearchParams uploadSearchParams = new UploadSearchParams(image);
viSearcher.uploadSearch(uploadSearchParams);
}
Please refer to the source code of camera demo app for the usage.
GET /colorsearch
Search by color solution is to search images with similar color by providing a color code. The color code should be in Hexadecimal and passed to the colorsearch service.
ColorSearchParams colorSearchParams = new ColorSearchParams("9b351b");
viSearch.colorSearch(colorSearchParams);
The search results are returned as a list of image names with required additional information. Use getImageList()
to get the list of images. The basic information returned about the image are image name. UseviSearch.cancelSearch()
to cancel a search, and handle the result by implementing the onSearchCanceled()
callback. If error occurs during the search, an error message will be returned and can be handled in viSearch.onSearchError(String error)
callback method.
@Override
public void onSearchResult(ResultList resultList) {
for (ImageResult imageResult : resultList.getImageList()) {
//Do something with each result image
...
}
}
@Override
public void onSearchError(String error) {
resultView.displayError(error);
}
@Override
public void onSearchCanceled() {
}
You can provide pagination parameters to control the paging of the image search results. by configuring the basic search parameters BaseSearchParams
. As the result is returned in a format of a list of images page by page, use setLimit
to set the number of results per page, setPage
to indicate the page number:
Name | Type | Description |
---|---|---|
page | Integer | Optional parameter to specify the page of results. The first page of result is 1. Defaults to 1. |
limit | Integer | Optional parameter to specify the result per page limit. Defaults to 10. |
BaseSearchParams baseSearchParams = new BaseSearchParams();
baseSearchParams.setLimit(20);
baseSearchParams.setPage(1);
ColorSearchParams colorSearchParams = new ColorSearchParams("3322ff");
colorSearchParams.setBaseSearchParams(baseSearchParams);
visearcher.colorSearch(colorSearchParams);
To retrieve metadata of your search results, provide a list of metadata keys as the fl
(field list) in the basic search property:
BaseSearchParams baseSearchParams = new BaseSearchParams();
List<String> fl = new ArrayList<>();
fl.add("price");
fl.add("brand");
baseSearchParams.setFl(fl);
UploadSearchParams uploadSearchParams = new UploadSearchParams(new Image("/path/to/image"));
uploadSearchParams.setBaseSearchParams(baseSearchParams);
To retrieve all metadata of your image results, specify get_all_fl
parameter and set it to true
:
BaseSearchParams baseSearchParams = new BaseSearchParams();
baseSearchParams.setGetAllFl(true);
UploadSearchParams uploadSearchParams = new UploadSearchParams(new Image("/path/to/image"));
uploadSearchParams.setBaseSearchParams(baseSearchParams);
In result callback you can read the metadata:
@Override
public void onSearchResult(ResultList resultList) {
for (ImageResult imageResult : resultList.getImageList()) {
Map<String, String> metaData = imageResult.getMetaData();
//Do something with the metadata
...
}
}
Only metadata of type string, int, and float can be retrieved from ViSearch. Metadata of type text is not available for retrieval.
To filter search results based on metadata values, provide a map of metadata key to filter value as the fq
(filter query) property:
// add fq param to specify the filtering criteria
Map<String, String> fq = new HashMap<String, String>();
// description is metadata type text
fq.put("description", "wingtips");
// price is metadata type float
fq.put("price", "0,199");
baseSearchParams.setFq(fq);
Querying syntax for each metadata type is listed in the following table:
Type | FQ |
---|---|
string | Metadata value must be exactly matched with the query value, e.g. "Vintage Wingtips" would not match "vintage wingtips" or "vintage" |
text | Metadata value will be indexed using full-text-search engine and supports fuzzy text matching, e.g. "A pair of high quality leather wingtips" would match any word in the phrase |
int | Metadata value can be either:
|
float | Metadata value can be either
|
ViSearch image search results are ranked in descending order i.e. from the highest scores to the lowest, ranging from 1.0 to 0.0. By default, the score for each result is not returned. You can turn on the score parameter to retrieve the scores for each image result:
// return scores for each image result, default is false
baseSearchParams.setScore(true);
IdSearchParams idSearchParams = new IdSearchParams("dress_1");
idSearchParams.setBaseSearchParams(baseSearchParams);
visearch.idSearch(idSearchParams);
...
@Override
public void onSearchResult(ResultList resultList) {
for (ImageResult imageResult : resultList.getImageList()) {
float score = imageResult.getScore();
//Do something with the score
...
}
}
If you need to restrict search results from a minimum score to a maximum score, specify the score_min and/or score_max parameters:
//set the threshold value
baseSearchParams.setScoreMin(0.5);
baseSearchParams.setScoreMax(0.8);
// only retrieve search results with scores between 0.5 and 0.8
IdSearchParams idSearchParams = new IdSearchParams("dress_1");
idSearchParams.setBaseSearchParams(baseSearchParams);
visearch.idSearch(idSearchParams);
With Automatic Object Recognition, ViSearch /uploadsearch API is smart to detect the objects present in the query image and suggest the best matched product type to run the search on.
You can turn on the feature in upload search by setting the API parameter "detection=all". We are now able to detect various types of fashion items, including Top
, Dress
, Bottom
, Shoe
, Bag
, Watch
and Indian Ethnic Wear
. The list is ever-expanding as we explore this feature for other categories.
Notice: This feature is currently available for fashion application type only. You will need to make sure your app type is configurated as "fashion" on ViSenze dashboard.
uploadSearchParams.setDetection("all");
You can use the Box parameter to restrict the image area [x1, y1, x2, y2] as the portion of your image to search for. When you input a box with 0 width and 0 height, eg. “box”:[574,224,574,224]. We will treat it as a point and detect the object over the current point.
You could also recognize objects from a paticular type on the uploaded query image through configuring the detection parameter to a specific product type as "detection={type}". Our API will run the search within that product type.
Sample request to detect bag
in an uploaded image:
uploadSearchParams.setDetection("bag");
The detected product types are listed in product_types
together with the match score and box area of the detected object. Multiple objects can be detected from the query image and they are ranked from the highest score to lowest. The full list of supported product types by our API will also be returned in product_types_list
.
ViSearch Android SDK provides methods to understand how your customer interact with the search results.
In addition, to improve subsequent search quality, it is recommended to send user actions when they interact with the results.
It is important that a unique device ID is provided for user action tracking. ViSearch Android SDK uses Google Adervertising ID as the default unique id. Add this tag to your <application>
in the AndroidManifest.xml
to use Google Play Service:
<meta-data android:name="com.google.android.gms.version" android:value="@integer/google_play_services_version">
In the case where Google Adervertising ID is not available, a server-generated UID will be returned. This UID is automatically stored with your app that integates with ViSearch Android SDK and will be refreshed only when the user uninstall your app.
User action can be sent in this way:
viSearch.track(new TrackParams().setAction(action).setImName(im_name).setReqid(reqid));
Name | Type | Description |
---|---|---|
action |
String | The the action type of this event. We are currently able to support "click". More actions will be supported in the future. |
im_name |
String | The imname of the image which the user has clicked on. im_name is the unique identifier of the indexed image. |
reqid |
String | The request id of the search request. This reqid can be obtained from all the search result:resultList.getTransId(); |