NativeScript Core

In this article
Not finding the help you need?

Using Async Data Fetch with RadAutoCompleteTextView

There are multiple scenarios where we need to represent data loaded from remote provider like a web service. Dynamic loading of items, based on the user input, could be quite useful when we are working with sets of data that are not available locally. In order to satisfy this scenario, the autocomplete provides you with API, which allows you asynchronously to load your data and then pass it to the control.

In next section we will discuss the process of working with remote data in details. For the purpose of the example we will use json data containing description of airports.

Implementation

Populating the autocomplete asynchronously is quite straightforward task. All you need is a promise which should handle the data fetch and return a collection of TokenModel objects. You should assign this promise to the loadSuggestionsAsync property of the RadAutoCompleteTextView object. The autocomplete executes this promise every time a symbol is typed and then generates suggestions based on the collection returned by the promise.

Create a page with RadAutoCompleteTextView control. Setup the control in a way that suits you and bind it to source collection, which in our case is called dataItems. Do not forget to provide a suggestionItemTemplate which will represent each suggestion.

<navigation:ExamplePage loaded="onPageLoaded" xmlns:navigation="navigation/example-page" xmlns:lv="nativescript-ui-autocomplete" xmlns="http://www.nativescript.org/tns.xsd">
    <StackLayout ios:backgroundColor="#CDCECE" padding="5">
        <Label text="Select destination:"></Label>
        <lv:RadAutoCompleteTextView id="autocomplete" items="{{ dataItems }}" suggestMode="Suggest" displayMode="Plain">
            <lv:RadAutoCompleteTextView.suggestionView>
                <lv:SuggestionView suggestionViewHeight="300">
                    <lv:SuggestionView.suggestionItemTemplate>
                        <StackLayout orientation="vertical" padding="10">
                            <Label text="{{ text }}"></Label>
                        </StackLayout>
                    </lv:SuggestionView.suggestionItemTemplate>
                </lv:SuggestionView>
            </lv:RadAutoCompleteTextView.suggestionView>
         </lv:RadAutoCompleteTextView>
    </StackLayout>
</navigation:ExamplePage>

Retrieve the RadAutoCompleteTextView object that was initialized in the xml file and set it's loadSuggestionsAsync property to e function which accepts one parameter (the typed text). In this function define a promise, load the remote data in it and then return the promise.

Later on the autocomplete will invoke the loadSuggestionsAsync function and when the promise is resolved, it will use the returned items to complete it's population.

export class ViewModel extends Observable {
    private autocomplete: RadAutoCompleteTextView;
    private jsonUrl = "https://raw.githubusercontent.com/NativeScript/nativescript-ui-samples/master/examples-data/airports.json";

    constructor(args) {
        super();
        const page = args.object;
        this.autocomplete = <RadAutoCompleteTextView>page.getViewById("autocomplete");
        let that = this;
        this.autocomplete.loadSuggestionsAsync = function (text) {
            const promise = new Promise((resolve, reject)   => {
                http.getJSON(that.jsonUrl).then((res: AirportJSON) => {
                    const airportsCollection = res.airports;
                    const items: Array<TokenModel> = new Array();
                    for (let i = 0; i < airportsCollection.length; i++) {
                        items.push(new TokenModel(airportsCollection[i].FIELD2, null));
                    }
                    resolve(items);
                }).catch((err) => {
                    const message = 'Error fetching remote data from ' + that.jsonUrl + ': ' + err.message;
                    console.log(message);
                    alert(message);
                    reject();
                });
            });

            return promise;
        };
    }

    get dataItems(): ObservableArray<TokenModel> {
        return this.get("_dataItems");
    }

    set dataItems(value: ObservableArray<TokenModel>) {
        this.set("_dataItems", value);
    }
}

References

Want to see this scenario in action? Check our SDK examples repo on GitHub. You will find this and many other practical examples with NativeScript UI.