0

我正在使用带有 TypeScript 的 Angular 1.5.x。为了访问远程 API,我使用了 restangular。作为总结,这是我的场景:

我的 API 有以下资源http://localhost:53384/api/timezones。向该 url 发送带有动词 GET 的请求会返回一个 JSON 数组:

[
   {
      "code":"Dateline Standard Time",
      "name":"(UTC-12:00) International Date Line West"
   },
   {
      "code":"UTC-11",
      "name":"(UTC-11:00) Coordinated Universal Time-11"
   },
   {
      "code":"Hawaiian Standard Time",
      "name":"(UTC-10:00) Hawaii"
   }
]

现在在我的带有 TypeScript 的客户端 AngularJs 应用程序中:

Restangular配置为restangularProvider:restangular.IProvider

restangularProvider.setBaseUrl("http://localhost:53384/api");

客户端中使用打字稿的 TimeZone 对象表示

module app.blocks {
    "use strict";

    export class TimeZone {
        public code: string;
        public name: string;
    }
}

工厂(restangular.IService)包装restangular所有“时区”资源

module app.services {
    factory.$inject = ["Restangular"];

    function factory(restangular: restangular.IService): restangular.IElement {
        return restangular.all("timezones");
    }

    angular
        .module("app.services")
        .factory("app.services.TimeZonesRestangular", factory);
}

使用TimeZonesRestangular包装其 restangular 功能并向以异步方式请求时区的任何人返回链式承诺的服务

module app.services {
    "use strict";

    export interface IStaticDataService {
        getTimeZones(): ng.IPromise<app.blocks.TimeZone[]>;
    }

    class StaticDataService implements IStaticDataService {
        constructor(private timeZonesRestangular: restangular.IElement) {
        }

        public getTimeZones(): ng.IPromise<blocks.TimeZone[]> {
            return this.timeZonesRestangular.getList()
                .then((timeZones: blocks.TimeZone[]) => {
                    return timeZones;
                }, (restangularError: any) => {
                    throw "Error retrieving time zones. Status: " + restangularError.status;
                });
        }
    }

    factory.$inject = ["app.services.TimeZonesRestangular"];

    function factory(timeZonesRestangular: restangular.IElement): IStaticDataService {
        return new StaticDataService(timeZonesRestangular);
    }

    angular
        .module("app.services")
        .factory("app.services.StaticDataService", factory);
}

最后在控制器中使用服务异步获取“时区”我有这个语句

//..other controller things not relevant for this sample
this.staticDataService.getTimeZones()
      .then((timeZones: blocks.TimeZone[]) => {
            this.timeZones = timeZones;
       });

有2个问题:

  1. restangular 的类型定义(我安装了tsd install restangular --resolve --save)告诉我 getTimeZones() 方法中的 successCallback 是 a promiseValue: any[],这很好,因为它确实是一个数组。我认为这将是一个 TimeZone[] 数组,并且打字稿可以正确编译,因为它接受any[],但是当调试时我看到 successCallback 承诺的值它不是 TimeZone[] 数组。它具有我期望的属性(codename),但它也有许多其他的东西。该数组中的对象如下所示(加上一些函数):

    {  
     "code":"Dateline Standard Time",
     "name":"(UTC-12:00) International Date Line West",
     "route":"timezones",
     "reqParams":null,
     "restangularized":true,
     "fromServer":true,
     "parentResource":null,
     "restangularCollection":false
    }
    

    根据https://github.com/mgonto/restangular/issues/150,看起来我的回复已经“重新调整了”。对于像我这样刚接触restangular的人的可怕描述。 我应该使用restangular类型定义中的什么接口来表示restangularized TimeZone[]数组?

  2. 有没有关于如何使用 TypeScript 实现类似功能的示例?

谢谢你。

4

1 回答 1

0

在进一步挖掘之后,我发现处理这个问题的最好方法是期待一个承诺的类型值restangular.ICollection(它继承自IServiceand Array<any>),这样我就可以像这样对响应进行去角化:

public getTimeZones(): ng.IPromise<blocks.TimeZone[]> {
            return this.timeZonesRestangular.getList()
                .then((restangularizedTimeZones: restangular.ICollection) => {
                    return restangularizedTimeZones.plain();
                }, (restangularError: any) => {
                    throw "Error retrieving time zones. Status: " + restangularError.status;
                });
        }

现在一切似乎都很好,回应确实是一个承诺TimeZone[]

于 2016-02-16T21:53:39.757 回答