- html - 出于某种原因,IE8 对我的 Sass 文件中继承的 html5 CSS 不友好?
- JMeter 在响应断言中使用 span 标签的问题
- html - 在 :hover and :active? 上具有不同效果的 CSS 动画
- html - 相对于居中的 html 内容固定的 CSS 重复背景?
我正在尝试创建一些包装 Angular2 装饰器的功能。我想简化向主机添加 CSS 类的过程,因此我创建了以下内容:
警告:不适用于 AOT 编译
type Constructor = {new(...args: any[]): {}};
export function AddCssClassToHost<T extends Constructor>(cssClass: string) {
return function (constructor: T) {
class Decorated extends constructor {
@HostBinding("class") cssClass = cssClass;
}
// Can't return an inline class, so we name it.
return Decorated;
};
}
我还希望能够创建另一个添加特定 CSS 类的装饰器。
/**
* Decorator to be used for components that are top level routes. Automatically adds the content-container class that is
* required so that the main content scrollbar plays nice with the header and the header won't scroll away.
*/
export function TopLevelRoutedComponent<T extends Constructor>(constructor: T) {
// This causes an error when called as a decorator
return AddCssClassToHost("content-container");
// Code below works, I'd like to avoid the duplication
// class Decorated extends constructor {
// @HostBinding("class") cssClass = "content-container";
// }
// return Decorated;
}
// Called like
@TopLevelRoutedComponent
@Component({
selector: "vcd-administration-navigation",
template: `
<div class="content-area">
<router-outlet></router-outlet>
</div>
<vcd-side-nav [navMenu]="navItems"></vcd-side-nav>`
})
export class AdminNavigationComponent {
navItems: NavItem[] = [{nameKey: "Multisite", routerLink: "multisite"}];
}
我得到的错误信息是
TS1238: Unable to resolve signature of class decorator when called as an expression.
Type '(constructor: Constructor) => { new (...args: any[]): AddCssClassToHost<Constructor>.Decorated; p...' is not assignable to type 'typeof AdminNavigationComponent'.
Type '(constructor: Constructor) => { new (...args: any[]): AddCssClassToHost<Constructor>.Decorated; p...' provides no match for the signature 'new (): AdminNavigationComponent'
我能够通过创建一个由两者调用的函数来解决这个问题
function wrapWithHostBindingClass<T extends Constructor>(constructor: T, cssClass: string) {
class Decorated extends constructor {
@HostBinding("class") cssClass = cssClass;
}
return Decorated; // Can't return an inline decorated class, name it.
}
export function AddCssClassToHost(cssClass: string) {
return function(constructor) {
return wrapWithHostBindingClass(constructor, cssClass);
};
}
export function TopLevelRoutedComponent(constructor) {
return wrapWithHostBindingClass(constructor, "content-container");
}
有没有一种方法可以使第一种样式起作用,而不需要辅助函数或复制代码?我意识到我的尝试不是很好而且没有意义,但我无法理解错误消息。
(某种)AOT 兼容的版本因为下面的要简单很多,不会导致AOT编译器崩溃。但是,请注意,如果使用 webpack 编译器,自定义装饰器将被删除,因此它仅在使用 ngc 编译时有效。
export function TopLevelRoutedComponent(constructor: Function) {
const propertyKey = "cssClass";
const className = "content-container";
const descriptor = {
enumerable: false,
configurable: false,
writable: false,
value: className
};
HostBinding("class")(constructor.prototype, propertyKey, descriptor);
Object.defineProperty(constructor.prototype, propertyKey, descriptor);
}
最佳答案
装饰器的类型签名中存在一些错误,无法满足类型检查器的要求。
为了修复它们,您必须了解装饰器 和装饰器工厂 之间的区别。
正如您可能怀疑的那样,装饰器工厂只不过是一个返回装饰器的函数。
当您想要自定义通过参数化应用的装饰器时,请使用装饰器工厂。
以您的代码为例,下面是一个装饰器工厂
export type Constructor<T = object> = new (...args: any[]) => T;
export function AddCssClassToHost<C extends Constructor>(cssClass: string) {
return function (Class: C) {
// TypeScript does not allow decorators on class expressions so we create a local
class Decorated extends Class {
@HostBinding("class") cssClass = cssClass;
}
return Decorated;
};
}
装饰器工厂采用一个参数来自定义它返回的装饰器使用的 css,并用(那是一口)替换目标。
但是现在我们想重新使用装饰器工厂来添加一些特定的css。这意味着我们需要一个普通的旧装饰器,而不是工厂。
由于装饰器工厂只返回我们需要的东西,所以我们可以直接调用它。
使用你的第二个例子,
export const TopLevelRoutedComponent = AddCssClassToHost("content-container");
现在我们遇到应用程序的错误
@TopLevelRoutedComponent
@Component({...})
export class AdminNavigationComponent {
navItems = [{nameKey: "Multisite", routerLink: "multisite"}];
}
我们必须考虑原因。
调用工厂函数实例化其类型参数。
我们需要一个创建通用装饰器的工厂,而不是返回非通用装饰器的通用装饰器工厂!
也就是说,TopLevelRoutedComponent
需要是通用的。
我们可以通过简单地重写我们原来的装饰器工厂,将类型参数移动到它返回的装饰器来做到这一点
export function AddCssClassToHost(cssClass: string) {
return function <C extends Constructor>(Class: C) {
// TypeScript does not allow decorators on class expressions so we create a local
class Decorated extends Class {
@HostBinding("class") cssClass = cssClass;
}
return Decorated;
};
}
这是一个live example
关于typescript - 如何在 TypeScript 的装饰器中重用装饰器,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/47994193/
我有这些 ViewModel:RecordViewModel、ComponentViewModel,其中 RecordViewModel 本质上是几个 ComponentViewModel 的容器。
我正在尝试为我的 Controller 设置一个装饰器。我的目的是在我的应用程序中的所有 Controller 中引入一些常见的行为。 我已将其配置为在 Angular 1.2.x 中工作,但从 1.
我想用角上的时间戳装饰我生成的所有 JFreeCharts。 JFreeChart 框架中是否有一种方法可以在生成图表后在图像上绘制? 编辑:请注意,这些图表是在后台线程中生成并通过 servlet
在grails应用程序中,我想用自定义数据装饰每个日志。当前的需要只是在日志消息前添加当前用户名 我对如何解决此问题有一些想法: -Adding custom field to log4j patte
我有一个form-el,它只是一个容器,必须将所有子元素包装在具有特定类的div 中。我希望允许 from-el 来包装它们,而不是在每个表单元素中重复这个 div 。我可以循环所有元素并将它们包装在
我遵循以下约定来装饰 Python 类中的某些方法。我想知道是否有一些更好的方法可以做到同样的事情。我的方法看起来当然不太好;对原始成员函数的调用看起来一点也不直观。 from threading i
我的 DTO 看起来像这样 public class SomeDTO { public string last_name{ get; set; } public string ac
这可能是一个远景,但 .NET 的灵 active 一直让我惊叹不已,所以开始吧。 我正在开发一个 MVC 应用程序,它需要在一组程序集中搜索派生自公共(public)基类的类类型。 即我有几个程序集
在我正在进行的项目中,我正在按照项目负责人的要求实现开发人员通知系统。它的工作方式是,如果发生前端错误,开发团队会收到一封错误电子邮件。 但是,在我当前的实现中,我似乎有以下循环依赖: $rootSc
我需要一些关于如何取消装饰和装饰 JInternalFrame 的帮助。我的框架类是这样的: package com; import java.awt.BorderLayout; import jav
假设我们有可能需要长时间运行的任务: public class LongRunningTask { public ReturnType doSomething() { ...
我正在尝试创建一个装饰器方法,它将一些默认的生命周期方法添加到 react 组件中。我的目标是向组件中添加一些默认功能,例如,所有组件都应该能够在 componentWillMount 上执行特定操作
我正在尝试将 DBUS 的异步方法调用与 Twisted 的 Deferred 相结合,但我在调整常用的 DBUS 服务方法装饰器来执行此操作时遇到了麻烦。 要使用 DBUS 异步回调方法,您需要:
是否可以设置表格中一行的背景颜色?当条件适用时,我需要突出显示一行。效果为 ...我可以在其中指定“字体”属性。 (我需要突出显示整行)。 最佳答案 您必须子类化 qooxdoo 默认行渲染器才能做到
我正在开发一个具有不同视角的基于 Java Swing 的应用程序。对于“主菜单”视角,我不希望装饰窗口 (JFrame),而在其他视角中,我确实希望装饰窗口。换句话说,我需要动态更 retrofit
我想做一些类似下面代码所示的事情: class foo { private: std::fstream* m_stream; public: foo(std::fstream* str
我的数据源提供了一个 ObservableList ,但是对于我的 ListView,我需要一个 ObservableList . A Warning基本上只是字符串的装饰器,添加一个 boolean
我一直在纠结于装饰+接口(interface)。假设我有以下“行为”接口(interface): interface IFlyable { void Fly();} interface ISwimma
有没有人为 iOS 6 UICollectionView 实现过装饰 View ?不可能 查找有关在网络上实现装饰 View 的任何教程。基本上在我的应用程序中,我有多个部分,我只想在每个部分后面显示
我有一个简单的 Controller ,例如: function MyController($scope, $http) { ... $http.post(url).success(f
我是一名优秀的程序员,十分优秀!