@banta/sdk 5.2.1 → 5.2.3

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
@@ -40,7 +40,7 @@ export class ChatMessageComponent {
40
40
  this._selected.next();
41
41
  }
42
42
  selectUser() {
43
- this._userSelected.next();
43
+ this._userSelected.next(this.message.user);
44
44
  }
45
45
  static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "17.3.9", ngImport: i0, type: ChatMessageComponent, deps: [], target: i0.ɵɵFactoryTarget.Component }); }
46
46
  static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "17.3.9", type: ChatMessageComponent, selector: "banta-chat-message", inputs: { message: "message" }, outputs: { userSelected: "userSelected", selected: "selected", reported: "reported", upvoted: "upvoted" }, ngImport: i0, template: "<div class=\"message-content\">\r\n <div class=\"user\" (click)=\"selectUser()\">\r\n <div class=\"avatar\" [style.background-image]=\"avatarForUser(message.user)\"></div>\r\n <label>{{message.user.username}}</label>\r\n </div>\r\n <div class=\"content\">\r\n <div (click)=\"select()\">\r\n {{message.message}}\r\n </div>\r\n <div class=\"status\">\r\n <div class=\"count-indicator\" *ngIf=\"message.likes > 0\">\r\n <mat-icon>star</mat-icon> {{message.likes}}\r\n </div>\r\n </div>\r\n </div>\r\n</div>\r\n<div class=\"actions\">\r\n <button mat-icon-button matTooltip=\"Upvote\" matTooltipPosition=\"below\" (click)=\"upvote()\">\r\n <mat-icon>thumb_up</mat-icon>\r\n </button>\r\n <button mat-icon-button matTooltip=\"Report\" matTooltipPosition=\"below\" (click)=\"report()\">\r\n <mat-icon>report</mat-icon>\r\n </button>\r\n</div>", styles: [":host{display:flex;flex-direction:row;align-items:center;padding:0 1em;background-color:#fff;color:#000;transition:.4s background-color ease-out}:host .message-content .content{color:#111}:host:hover{background-color:#ddd}:host.highlight{background:#00121b}:host.highlight:hover{background:#01324d}:host:nth-child(2n){background-color:#eee}:host:nth-child(2n):hover{background:#ddd}:host:nth-child(2n) .message-content .content{color:#222}:host:nth-child(2n).highlight{background:#001a2a}:host:nth-child(2n).highlight:hover{background:#002b44}:host .message-content{display:flex;flex-direction:row;flex-grow:1;align-items:center}:host .message-content .content{display:flex;flex-direction:row;padding:5px 0}:host .message-content .content .status{display:flex;flex-direction:row;align-items:center;margin-left:1em}:host .message-content .content .status mat-icon{margin-left:.5em}:host .user{color:#999;font-weight:400;text-align:right;margin-right:.25em;flex-shrink:0;display:flex;align-items:center}:host .user .avatar{background-position:center;background-size:cover;background-color:#333;border-radius:100%;flex-shrink:0;flex-grow:0;margin-right:1em;width:2em;height:2em}:host .user:after{content:\":\";margin-right:1em}:host .content{flex-grow:1}:host .actions{flex-shrink:0;white-space:nowrap;opacity:0;transition:.4s opacity ease-out}:host:hover .actions{opacity:1}.count-indicator{white-space:nowrap}.count-indicator mat-icon{vertical-align:bottom}:host-context(.mat-dark-theme){background-color:#000;color:#fff}:host-context(.mat-dark-theme) .message-content .content{color:#ddd}:host-context(.mat-dark-theme):hover{background-color:#111}:host-context(.mat-dark-theme):nth-child(2n).highlight{background:#001a2a}:host-context(.mat-dark-theme):nth-child(2n).highlight:hover{background:#002b44}:host-context(.mat-dark-theme):nth-child(2n):hover{background-color:#111}:host-context(.mat-dark-theme):nth-child(2n){background-color:#080808}:host-context(.mat-dark-theme):nth-child(2n) .message-content .content{color:#eee}label{margin:0}\n"], dependencies: [{ kind: "directive", type: i1.NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }, { kind: "component", type: i2.MatIcon, selector: "mat-icon", inputs: ["color", "inline", "svgIcon", "fontSet", "fontIcon"], exportAs: ["matIcon"] }, { kind: "component", type: i3.MatIconButton, selector: "button[mat-icon-button]", exportAs: ["matButton"] }] }); }
@@ -59,4 +59,4 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "17.3.9", ngImpor
59
59
  }], upvoted: [{
60
60
  type: Output
61
61
  }] } });
62
- //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiY2hhdC1tZXNzYWdlLmNvbXBvbmVudC5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uLy4uLy4uLy4uLy4uL3Byb2plY3RzL3Nkay9zcmMvbGliL2NoYXQvY2hhdC1tZXNzYWdlL2NoYXQtbWVzc2FnZS5jb21wb25lbnQudHMiLCIuLi8uLi8uLi8uLi8uLi8uLi9wcm9qZWN0cy9zZGsvc3JjL2xpYi9jaGF0L2NoYXQtbWVzc2FnZS9jaGF0LW1lc3NhZ2UuY29tcG9uZW50Lmh0bWwiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IkFBQUEsT0FBTyxFQUFFLFNBQVMsRUFBRSxLQUFLLEVBQUUsTUFBTSxFQUFFLE1BQU0sZUFBZSxDQUFDO0FBRXpELE9BQU8sRUFBRSxPQUFPLEVBQUUsTUFBTSxNQUFNLENBQUM7Ozs7O0FBTy9CLE1BQU0sT0FBTyxvQkFBb0I7SUFMakM7UUFNWSxjQUFTLEdBQUcsSUFBSSxPQUFPLEVBQVEsQ0FBQztRQUNoQyxjQUFTLEdBQUcsSUFBSSxPQUFPLEVBQVEsQ0FBQztRQUNoQyxhQUFRLEdBQUcsSUFBSSxPQUFPLEVBQVEsQ0FBQztRQUMvQixrQkFBYSxHQUFHLElBQUksT0FBTyxFQUFRLENBQUM7S0FrRC9DO0lBN0NHLElBQ0ksWUFBWTtRQUNaLE9BQU8sSUFBSSxDQUFDLGFBQWEsQ0FBQztJQUM5QixDQUFDO0lBRUQsSUFDSSxRQUFRO1FBQ1IsT0FBTyxJQUFJLENBQUMsU0FBUyxDQUFDO0lBQzFCLENBQUM7SUFFRCxJQUNJLFFBQVE7UUFDUixPQUFPLElBQUksQ0FBQyxTQUFTLENBQUM7SUFDMUIsQ0FBQztJQUVELElBQ0ksT0FBTztRQUNQLE9BQU8sSUFBSSxDQUFDLFFBQVEsQ0FBQztJQUN6QixDQUFDO0lBRUQsYUFBYSxDQUFDLElBQVc7UUFDckIsSUFBSSxJQUFJLElBQUksSUFBSSxDQUFDLFNBQVMsRUFBRSxDQUFDO1lBQ3pCLElBQUksR0FBRyxHQUFHLElBQUksQ0FBQyxTQUFTLENBQUM7WUFDekIsT0FBTyxPQUFPLEdBQUcsR0FBRyxDQUFDO1FBQ3pCLENBQUM7UUFFRCxPQUFPLElBQUksQ0FBQztJQUNoQixDQUFDO0lBRUQsTUFBTTtRQUNGLElBQUksQ0FBQyxRQUFRLENBQUMsSUFBSSxFQUFFLENBQUM7SUFDekIsQ0FBQztJQUVELE1BQU07UUFDRixJQUFJLENBQUMsU0FBUyxDQUFDLElBQUksRUFBRSxDQUFDO0lBQzFCLENBQUM7SUFFRCxNQUFNO1FBQ0YsSUFBSSxDQUFDLFNBQVMsQ0FBQyxJQUFJLEVBQUUsQ0FBQztJQUMxQixDQUFDO0lBRUQsVUFBVTtRQUNOLElBQUksQ0FBQyxhQUFhLENBQUMsSUFBSSxFQUFFLENBQUM7SUFDOUIsQ0FBQzs4R0FwRFEsb0JBQW9CO2tHQUFwQixvQkFBb0IscU1DVGpDLHM4QkF1Qk07OzJGRGRPLG9CQUFvQjtrQkFMaEMsU0FBUzsrQkFDSSxvQkFBb0I7OEJBVzlCLE9BQU87c0JBRE4sS0FBSztnQkFJRixZQUFZO3NCQURmLE1BQU07Z0JBTUgsUUFBUTtzQkFEWCxNQUFNO2dCQU1ILFFBQVE7c0JBRFgsTUFBTTtnQkFNSCxPQUFPO3NCQURWLE1BQU0iLCJzb3VyY2VzQ29udGVudCI6WyJpbXBvcnQgeyBDb21wb25lbnQsIElucHV0LCBPdXRwdXQgfSBmcm9tIFwiQGFuZ3VsYXIvY29yZVwiO1xyXG5pbXBvcnQgeyBVc2VyLCBDaGF0TWVzc2FnZSB9IGZyb20gJ0BiYW50YS9jb21tb24nO1xyXG5pbXBvcnQgeyBTdWJqZWN0IH0gZnJvbSAncnhqcyc7XHJcblxyXG5AQ29tcG9uZW50KHtcclxuICAgIHNlbGVjdG9yOiAnYmFudGEtY2hhdC1tZXNzYWdlJyxcclxuICAgIHRlbXBsYXRlVXJsOiAnLi9jaGF0LW1lc3NhZ2UuY29tcG9uZW50Lmh0bWwnLFxyXG4gICAgc3R5bGVVcmxzOiBbJy4vY2hhdC1tZXNzYWdlLmNvbXBvbmVudC5zY3NzJ11cclxufSlcclxuZXhwb3J0IGNsYXNzIENoYXRNZXNzYWdlQ29tcG9uZW50IHtcclxuICAgIHByaXZhdGUgX3NlbGVjdGVkID0gbmV3IFN1YmplY3Q8dm9pZD4oKTtcclxuICAgIHByaXZhdGUgX3JlcG9ydGVkID0gbmV3IFN1YmplY3Q8dm9pZD4oKTtcclxuICAgIHByaXZhdGUgX3Vwdm90ZWQgPSBuZXcgU3ViamVjdDx2b2lkPigpO1xyXG4gICAgcHJpdmF0ZSBfdXNlclNlbGVjdGVkID0gbmV3IFN1YmplY3Q8VXNlcj4oKTtcclxuXHJcbiAgICBASW5wdXQoKVxyXG4gICAgbWVzc2FnZSA6IENoYXRNZXNzYWdlO1xyXG4gICAgXHJcbiAgICBAT3V0cHV0KCkgXHJcbiAgICBnZXQgdXNlclNlbGVjdGVkKCkge1xyXG4gICAgICAgIHJldHVybiB0aGlzLl91c2VyU2VsZWN0ZWQ7XHJcbiAgICB9XHJcblxyXG4gICAgQE91dHB1dCgpXHJcbiAgICBnZXQgc2VsZWN0ZWQoKSB7XHJcbiAgICAgICAgcmV0dXJuIHRoaXMuX3NlbGVjdGVkO1xyXG4gICAgfVxyXG5cclxuICAgIEBPdXRwdXQoKVxyXG4gICAgZ2V0IHJlcG9ydGVkKCkge1xyXG4gICAgICAgIHJldHVybiB0aGlzLl9yZXBvcnRlZDtcclxuICAgIH1cclxuXHJcbiAgICBAT3V0cHV0KClcclxuICAgIGdldCB1cHZvdGVkKCkge1xyXG4gICAgICAgIHJldHVybiB0aGlzLl91cHZvdGVkO1xyXG4gICAgfVxyXG5cclxuICAgIGF2YXRhckZvclVzZXIodXNlciA6IFVzZXIpIHtcclxuICAgICAgICBpZiAodXNlciAmJiB1c2VyLmF2YXRhclVybCkge1xyXG4gICAgICAgICAgICBsZXQgdXJsID0gdXNlci5hdmF0YXJVcmw7XHJcbiAgICAgICAgICAgIHJldHVybiBgdXJsKCR7dXJsfSlgO1xyXG4gICAgICAgIH1cclxuXHJcbiAgICAgICAgcmV0dXJuIG51bGw7XHJcbiAgICB9XHJcbiAgICBcclxuICAgIHVwdm90ZSgpIHtcclxuICAgICAgICB0aGlzLl91cHZvdGVkLm5leHQoKTtcclxuICAgIH1cclxuXHJcbiAgICByZXBvcnQoKSB7XHJcbiAgICAgICAgdGhpcy5fcmVwb3J0ZWQubmV4dCgpO1xyXG4gICAgfVxyXG5cclxuICAgIHNlbGVjdCgpIHtcclxuICAgICAgICB0aGlzLl9zZWxlY3RlZC5uZXh0KCk7XHJcbiAgICB9XHJcbiAgICBcclxuICAgIHNlbGVjdFVzZXIoKSB7XHJcbiAgICAgICAgdGhpcy5fdXNlclNlbGVjdGVkLm5leHQoKTtcclxuICAgIH1cclxuXHJcbn0iLCI8ZGl2IGNsYXNzPVwibWVzc2FnZS1jb250ZW50XCI+XHJcbiAgICA8ZGl2IGNsYXNzPVwidXNlclwiIChjbGljayk9XCJzZWxlY3RVc2VyKClcIj5cclxuICAgICAgICA8ZGl2IGNsYXNzPVwiYXZhdGFyXCIgW3N0eWxlLmJhY2tncm91bmQtaW1hZ2VdPVwiYXZhdGFyRm9yVXNlcihtZXNzYWdlLnVzZXIpXCI+PC9kaXY+XHJcbiAgICAgICAgPGxhYmVsPnt7bWVzc2FnZS51c2VyLnVzZXJuYW1lfX08L2xhYmVsPlxyXG4gICAgPC9kaXY+XHJcbiAgICA8ZGl2IGNsYXNzPVwiY29udGVudFwiPlxyXG4gICAgICAgIDxkaXYgKGNsaWNrKT1cInNlbGVjdCgpXCI+XHJcbiAgICAgICAgICAgIHt7bWVzc2FnZS5tZXNzYWdlfX1cclxuICAgICAgICA8L2Rpdj5cclxuICAgICAgICA8ZGl2IGNsYXNzPVwic3RhdHVzXCI+XHJcbiAgICAgICAgICAgIDxkaXYgY2xhc3M9XCJjb3VudC1pbmRpY2F0b3JcIiAqbmdJZj1cIm1lc3NhZ2UubGlrZXMgPiAwXCI+XHJcbiAgICAgICAgICAgICAgICA8bWF0LWljb24+c3RhcjwvbWF0LWljb24+IHt7bWVzc2FnZS5saWtlc319XHJcbiAgICAgICAgICAgIDwvZGl2PlxyXG4gICAgICAgIDwvZGl2PlxyXG4gICAgPC9kaXY+XHJcbjwvZGl2PlxyXG48ZGl2IGNsYXNzPVwiYWN0aW9uc1wiPlxyXG4gICAgPGJ1dHRvbiBtYXQtaWNvbi1idXR0b24gbWF0VG9vbHRpcD1cIlVwdm90ZVwiIG1hdFRvb2x0aXBQb3NpdGlvbj1cImJlbG93XCIgKGNsaWNrKT1cInVwdm90ZSgpXCI+XHJcbiAgICAgICAgPG1hdC1pY29uPnRodW1iX3VwPC9tYXQtaWNvbj5cclxuICAgIDwvYnV0dG9uPlxyXG4gICAgPGJ1dHRvbiBtYXQtaWNvbi1idXR0b24gbWF0VG9vbHRpcD1cIlJlcG9ydFwiIG1hdFRvb2x0aXBQb3NpdGlvbj1cImJlbG93XCIgKGNsaWNrKT1cInJlcG9ydCgpXCI+XHJcbiAgICAgICAgPG1hdC1pY29uPnJlcG9ydDwvbWF0LWljb24+XHJcbiAgICA8L2J1dHRvbj5cclxuPC9kaXY+Il19
62
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiY2hhdC1tZXNzYWdlLmNvbXBvbmVudC5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uLy4uLy4uLy4uLy4uL3Byb2plY3RzL3Nkay9zcmMvbGliL2NoYXQvY2hhdC1tZXNzYWdlL2NoYXQtbWVzc2FnZS5jb21wb25lbnQudHMiLCIuLi8uLi8uLi8uLi8uLi8uLi9wcm9qZWN0cy9zZGsvc3JjL2xpYi9jaGF0L2NoYXQtbWVzc2FnZS9jaGF0LW1lc3NhZ2UuY29tcG9uZW50Lmh0bWwiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IkFBQUEsT0FBTyxFQUFFLFNBQVMsRUFBRSxLQUFLLEVBQUUsTUFBTSxFQUFFLE1BQU0sZUFBZSxDQUFDO0FBRXpELE9BQU8sRUFBRSxPQUFPLEVBQUUsTUFBTSxNQUFNLENBQUM7Ozs7O0FBTy9CLE1BQU0sT0FBTyxvQkFBb0I7SUFMakM7UUFNWSxjQUFTLEdBQUcsSUFBSSxPQUFPLEVBQVEsQ0FBQztRQUNoQyxjQUFTLEdBQUcsSUFBSSxPQUFPLEVBQVEsQ0FBQztRQUNoQyxhQUFRLEdBQUcsSUFBSSxPQUFPLEVBQVEsQ0FBQztRQUMvQixrQkFBYSxHQUFHLElBQUksT0FBTyxFQUFRLENBQUM7S0FrRC9DO0lBN0NHLElBQ0ksWUFBWTtRQUNaLE9BQU8sSUFBSSxDQUFDLGFBQWEsQ0FBQztJQUM5QixDQUFDO0lBRUQsSUFDSSxRQUFRO1FBQ1IsT0FBTyxJQUFJLENBQUMsU0FBUyxDQUFDO0lBQzFCLENBQUM7SUFFRCxJQUNJLFFBQVE7UUFDUixPQUFPLElBQUksQ0FBQyxTQUFTLENBQUM7SUFDMUIsQ0FBQztJQUVELElBQ0ksT0FBTztRQUNQLE9BQU8sSUFBSSxDQUFDLFFBQVEsQ0FBQztJQUN6QixDQUFDO0lBRUQsYUFBYSxDQUFDLElBQVc7UUFDckIsSUFBSSxJQUFJLElBQUksSUFBSSxDQUFDLFNBQVMsRUFBRSxDQUFDO1lBQ3pCLElBQUksR0FBRyxHQUFHLElBQUksQ0FBQyxTQUFTLENBQUM7WUFDekIsT0FBTyxPQUFPLEdBQUcsR0FBRyxDQUFDO1FBQ3pCLENBQUM7UUFFRCxPQUFPLElBQUksQ0FBQztJQUNoQixDQUFDO0lBRUQsTUFBTTtRQUNGLElBQUksQ0FBQyxRQUFRLENBQUMsSUFBSSxFQUFFLENBQUM7SUFDekIsQ0FBQztJQUVELE1BQU07UUFDRixJQUFJLENBQUMsU0FBUyxDQUFDLElBQUksRUFBRSxDQUFDO0lBQzFCLENBQUM7SUFFRCxNQUFNO1FBQ0YsSUFBSSxDQUFDLFNBQVMsQ0FBQyxJQUFJLEVBQUUsQ0FBQztJQUMxQixDQUFDO0lBRUQsVUFBVTtRQUNOLElBQUksQ0FBQyxhQUFhLENBQUMsSUFBSSxDQUFDLElBQUksQ0FBQyxPQUFPLENBQUMsSUFBSSxDQUFDLENBQUM7SUFDL0MsQ0FBQzs4R0FwRFEsb0JBQW9CO2tHQUFwQixvQkFBb0IscU1DVGpDLHM4QkF1Qk07OzJGRGRPLG9CQUFvQjtrQkFMaEMsU0FBUzsrQkFDSSxvQkFBb0I7OEJBVzlCLE9BQU87c0JBRE4sS0FBSztnQkFJRixZQUFZO3NCQURmLE1BQU07Z0JBTUgsUUFBUTtzQkFEWCxNQUFNO2dCQU1ILFFBQVE7c0JBRFgsTUFBTTtnQkFNSCxPQUFPO3NCQURWLE1BQU0iLCJzb3VyY2VzQ29udGVudCI6WyJpbXBvcnQgeyBDb21wb25lbnQsIElucHV0LCBPdXRwdXQgfSBmcm9tIFwiQGFuZ3VsYXIvY29yZVwiO1xyXG5pbXBvcnQgeyBVc2VyLCBDaGF0TWVzc2FnZSB9IGZyb20gJ0BiYW50YS9jb21tb24nO1xyXG5pbXBvcnQgeyBTdWJqZWN0IH0gZnJvbSAncnhqcyc7XHJcblxyXG5AQ29tcG9uZW50KHtcclxuICAgIHNlbGVjdG9yOiAnYmFudGEtY2hhdC1tZXNzYWdlJyxcclxuICAgIHRlbXBsYXRlVXJsOiAnLi9jaGF0LW1lc3NhZ2UuY29tcG9uZW50Lmh0bWwnLFxyXG4gICAgc3R5bGVVcmxzOiBbJy4vY2hhdC1tZXNzYWdlLmNvbXBvbmVudC5zY3NzJ11cclxufSlcclxuZXhwb3J0IGNsYXNzIENoYXRNZXNzYWdlQ29tcG9uZW50IHtcclxuICAgIHByaXZhdGUgX3NlbGVjdGVkID0gbmV3IFN1YmplY3Q8dm9pZD4oKTtcclxuICAgIHByaXZhdGUgX3JlcG9ydGVkID0gbmV3IFN1YmplY3Q8dm9pZD4oKTtcclxuICAgIHByaXZhdGUgX3Vwdm90ZWQgPSBuZXcgU3ViamVjdDx2b2lkPigpO1xyXG4gICAgcHJpdmF0ZSBfdXNlclNlbGVjdGVkID0gbmV3IFN1YmplY3Q8VXNlcj4oKTtcclxuXHJcbiAgICBASW5wdXQoKVxyXG4gICAgbWVzc2FnZSA6IENoYXRNZXNzYWdlO1xyXG4gICAgXHJcbiAgICBAT3V0cHV0KCkgXHJcbiAgICBnZXQgdXNlclNlbGVjdGVkKCkge1xyXG4gICAgICAgIHJldHVybiB0aGlzLl91c2VyU2VsZWN0ZWQ7XHJcbiAgICB9XHJcblxyXG4gICAgQE91dHB1dCgpXHJcbiAgICBnZXQgc2VsZWN0ZWQoKSB7XHJcbiAgICAgICAgcmV0dXJuIHRoaXMuX3NlbGVjdGVkO1xyXG4gICAgfVxyXG5cclxuICAgIEBPdXRwdXQoKVxyXG4gICAgZ2V0IHJlcG9ydGVkKCkge1xyXG4gICAgICAgIHJldHVybiB0aGlzLl9yZXBvcnRlZDtcclxuICAgIH1cclxuXHJcbiAgICBAT3V0cHV0KClcclxuICAgIGdldCB1cHZvdGVkKCkge1xyXG4gICAgICAgIHJldHVybiB0aGlzLl91cHZvdGVkO1xyXG4gICAgfVxyXG5cclxuICAgIGF2YXRhckZvclVzZXIodXNlciA6IFVzZXIpIHtcclxuICAgICAgICBpZiAodXNlciAmJiB1c2VyLmF2YXRhclVybCkge1xyXG4gICAgICAgICAgICBsZXQgdXJsID0gdXNlci5hdmF0YXJVcmw7XHJcbiAgICAgICAgICAgIHJldHVybiBgdXJsKCR7dXJsfSlgO1xyXG4gICAgICAgIH1cclxuXHJcbiAgICAgICAgcmV0dXJuIG51bGw7XHJcbiAgICB9XHJcbiAgICBcclxuICAgIHVwdm90ZSgpIHtcclxuICAgICAgICB0aGlzLl91cHZvdGVkLm5leHQoKTtcclxuICAgIH1cclxuXHJcbiAgICByZXBvcnQoKSB7XHJcbiAgICAgICAgdGhpcy5fcmVwb3J0ZWQubmV4dCgpO1xyXG4gICAgfVxyXG5cclxuICAgIHNlbGVjdCgpIHtcclxuICAgICAgICB0aGlzLl9zZWxlY3RlZC5uZXh0KCk7XHJcbiAgICB9XHJcbiAgICBcclxuICAgIHNlbGVjdFVzZXIoKSB7XHJcbiAgICAgICAgdGhpcy5fdXNlclNlbGVjdGVkLm5leHQodGhpcy5tZXNzYWdlLnVzZXIpO1xyXG4gICAgfVxyXG5cclxufSIsIjxkaXYgY2xhc3M9XCJtZXNzYWdlLWNvbnRlbnRcIj5cclxuICAgIDxkaXYgY2xhc3M9XCJ1c2VyXCIgKGNsaWNrKT1cInNlbGVjdFVzZXIoKVwiPlxyXG4gICAgICAgIDxkaXYgY2xhc3M9XCJhdmF0YXJcIiBbc3R5bGUuYmFja2dyb3VuZC1pbWFnZV09XCJhdmF0YXJGb3JVc2VyKG1lc3NhZ2UudXNlcilcIj48L2Rpdj5cclxuICAgICAgICA8bGFiZWw+e3ttZXNzYWdlLnVzZXIudXNlcm5hbWV9fTwvbGFiZWw+XHJcbiAgICA8L2Rpdj5cclxuICAgIDxkaXYgY2xhc3M9XCJjb250ZW50XCI+XHJcbiAgICAgICAgPGRpdiAoY2xpY2spPVwic2VsZWN0KClcIj5cclxuICAgICAgICAgICAge3ttZXNzYWdlLm1lc3NhZ2V9fVxyXG4gICAgICAgIDwvZGl2PlxyXG4gICAgICAgIDxkaXYgY2xhc3M9XCJzdGF0dXNcIj5cclxuICAgICAgICAgICAgPGRpdiBjbGFzcz1cImNvdW50LWluZGljYXRvclwiICpuZ0lmPVwibWVzc2FnZS5saWtlcyA+IDBcIj5cclxuICAgICAgICAgICAgICAgIDxtYXQtaWNvbj5zdGFyPC9tYXQtaWNvbj4ge3ttZXNzYWdlLmxpa2VzfX1cclxuICAgICAgICAgICAgPC9kaXY+XHJcbiAgICAgICAgPC9kaXY+XHJcbiAgICA8L2Rpdj5cclxuPC9kaXY+XHJcbjxkaXYgY2xhc3M9XCJhY3Rpb25zXCI+XHJcbiAgICA8YnV0dG9uIG1hdC1pY29uLWJ1dHRvbiBtYXRUb29sdGlwPVwiVXB2b3RlXCIgbWF0VG9vbHRpcFBvc2l0aW9uPVwiYmVsb3dcIiAoY2xpY2spPVwidXB2b3RlKClcIj5cclxuICAgICAgICA8bWF0LWljb24+dGh1bWJfdXA8L21hdC1pY29uPlxyXG4gICAgPC9idXR0b24+XHJcbiAgICA8YnV0dG9uIG1hdC1pY29uLWJ1dHRvbiBtYXRUb29sdGlwPVwiUmVwb3J0XCIgbWF0VG9vbHRpcFBvc2l0aW9uPVwiYmVsb3dcIiAoY2xpY2spPVwicmVwb3J0KClcIj5cclxuICAgICAgICA8bWF0LWljb24+cmVwb3J0PC9tYXQtaWNvbj5cclxuICAgIDwvYnV0dG9uPlxyXG48L2Rpdj4iXX0=
@@ -36,6 +36,8 @@ export class BantaAttachmentComponent {
36
36
  window['twttr']?.widgets.load();
37
37
  }
38
38
  loadPlatformSpecific() {
39
+ if (typeof window === 'undefined')
40
+ return;
39
41
  if (this._attachment?.type === 'tweet') {
40
42
  if (!TWITTER_LOADED && document.querySelector('script[src="https://platform.twitter.com/widgets.js"]'))
41
43
  TWITTER_LOADED = true;
@@ -123,4 +125,4 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "17.3.9", ngImpor
123
125
  type: HostBinding,
124
126
  args: ['class.loading']
125
127
  }] } });
126
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"attachment.component.js","sourceRoot":"","sources":["../../../../../../projects/sdk/src/lib/common/attachment/attachment.component.ts","../../../../../../projects/sdk/src/lib/common/attachment/attachment.component.html"],"names":[],"mappings":"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","sourcesContent":["import { Component, ElementRef, HostBinding, Input, Output } from \"@angular/core\";\r\nimport { ChatMessageAttachment } from \"@banta/common\";\r\nimport { Subject } from \"rxjs\";\r\n\r\nlet TWITTER_LOADED = false;\r\n\r\n@Component({\r\n    selector: 'banta-attachment',\r\n    templateUrl: './attachment.component.html',\r\n    styleUrls: ['./attachment.component.scss']\r\n})\r\nexport class BantaAttachmentComponent {\r\n    constructor(\r\n        private elementRef: ElementRef<HTMLElement>\r\n    ) {\r\n\r\n    }\r\n\r\n    private _attachment: ChatMessageAttachment;\r\n    @Input() get attachment() { return this._attachment; }\r\n    set attachment(value) {\r\n        this._attachment = value;\r\n        this.checkLoad();\r\n        this.loadPlatformSpecific();\r\n    }\r\n    \r\n    @Input() loading = false;\r\n    @Input() editing = false;\r\n    @Input() loadingMessage: string = 'Please wait...';\r\n    @Input() error = false;\r\n    @Input() errorMessage: string = 'An error has occurred';\r\n    @Output() removed = new Subject<void>();\r\n    @Output() activated = new Subject<void>();\r\n    @Output() loaded = new Subject<void>();\r\n\r\n    private _viewLoaded = false;\r\n    ngAfterViewInit() {\r\n        this._viewLoaded = true;\r\n        this.checkLoad();\r\n    }\r\n\r\n    private loadTwitterWidgets() {\r\n        if (typeof window !== 'undefined')\r\n            window['twttr']?.widgets.load();\r\n    }\r\n\r\n    private loadPlatformSpecific() {\r\n        if (this._attachment?.type === 'tweet') {\r\n            if (!TWITTER_LOADED && document.querySelector('script[src=\"https://platform.twitter.com/widgets.js\"]'))\r\n                TWITTER_LOADED = true;\r\n\r\n            if (typeof window !== 'undefined' && !TWITTER_LOADED) {\r\n                TWITTER_LOADED = true;\r\n                let script = document.createElement('script');\r\n                script.src = 'https://platform.twitter.com/widgets.js';\r\n                script.async = true;\r\n                script.addEventListener('load', () => setTimeout(() => this.loadTwitterWidgets()));\r\n                document.body.appendChild(script);\r\n            } else {\r\n                setTimeout(() => this.loadTwitterWidgets());\r\n            }\r\n        }\r\n    }\r\n    \r\n    private checkLoad() {\r\n        if (!this._attachment || !this._viewLoaded || !this.elementRef?.nativeElement)\r\n            return;\r\n\r\n        if (typeof window === 'undefined')\r\n            this.loaded.next();\r\n        else\r\n            setTimeout(() => this.loaded.next(), 250);\r\n    }\r\n\r\n    activate() {\r\n        this.activated.next();\r\n    }\r\n\r\n    remove() {\r\n        this.removed.next();\r\n    }\r\n\r\n    get isError() {\r\n        return this.error || this.attachment?.transientState?.error;\r\n    }\r\n\r\n    get theErrorMessage() {\r\n        return this.errorMessage || this.attachment?.transientState?.errorMessage;\r\n    }\r\n\r\n    @HostBinding('class.loading')\r\n    get isLoading() {\r\n        return this.editing && (\r\n            this.loading || !this.attachment || this.attachment.transientState?.loading \r\n            || !this.attachment.url\r\n        );\r\n    }\r\n\r\n    get isImageAttachment() {\r\n        return this.attachment?.type?.startsWith('image/');\r\n    }\r\n\r\n    get hasFrame() {\r\n        if (!this.attachment)\r\n            return false;\r\n\r\n        return this.attachment.type === 'iframe' || (\r\n            this.attachment.type === 'card' \r\n            && this.attachment.card?.player\r\n        );\r\n    }\r\n\r\n    get frameUrl() {\r\n        if (!this.attachment)\r\n            return undefined;\r\n        \r\n        if (this.attachment.type === 'iframe') {\r\n            return this.attachment.url;\r\n        } else if (this.attachment.type === 'card') {\r\n            return this.attachment.card.player;\r\n        }\r\n    }\r\n}","<button type=\"button\" (click)=\"remove()\" mat-mini-fab color=\"primary\" class=\"remove-button\" *ngIf=\"editing\">\r\n    <mat-icon>close</mat-icon>\r\n</button>\r\n\r\n<ng-container *ngIf=\"isError\">\r\n    <mat-icon class=\"error\">close</mat-icon>\r\n    <em class=\"error\">{{theErrorMessage}}</em>\r\n</ng-container>\r\n<ng-container *ngIf=\"!isError\">\r\n    <ng-container *ngIf=\"isLoading\">\r\n        <mat-spinner></mat-spinner>\r\n        <em>{{loadingMessage}}</em>\r\n    </ng-container>\r\n    <ng-container *ngIf=\"!isLoading && attachment\">\r\n        <iframe *ngIf=\"hasFrame\"\r\n            sandbox=\"allow-scripts allow-popups allow-same-origin allow-presentation\" \r\n            [src]=\"frameUrl | trustResourceUrl\"></iframe>\r\n        <a *ngIf=\"attachment.type === 'card'\" class=\"card-attachment\" [href]=\"attachment.url\" target=\"_blank\" [class.has-image]=\"attachment.card.image\">\r\n            <img \r\n                *ngIf=\"attachment.card.image\"\r\n                class=\"thumbnail\" \r\n                [src]=\"attachment.card.image\"\r\n                />\r\n            <div class=\"description\">\r\n                <h1>{{attachment.card.title}}</h1>\r\n                <div class=\"summary\">\r\n                    {{attachment.card.description}}\r\n                </div>\r\n                <cite>{{attachment.card.url}}</cite>\r\n            </div>\r\n        </a>\r\n        <a class=\"image-attachment\" *ngIf=\"isImageAttachment && attachment.url\" href=\"javascript:;\" (click)=\"activate()\">\r\n            <img [src]=\"attachment.url\" alt=\"Image Attachment\">\r\n        </a>\r\n        <blockquote *ngIf=\"attachment.type === 'tweet'\" \r\n            class=\"twitter-tweet\">\r\n                <p lang=\"en\" dir=\"ltr\"></p>\r\n                <a [href]=\"attachment.url\"></a>\r\n        </blockquote>\r\n    </ng-container>\r\n</ng-container>"]}
128
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"attachment.component.js","sourceRoot":"","sources":["../../../../../../projects/sdk/src/lib/common/attachment/attachment.component.ts","../../../../../../projects/sdk/src/lib/common/attachment/attachment.component.html"],"names":[],"mappings":"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","sourcesContent":["import { Component, ElementRef, HostBinding, Input, Output } from \"@angular/core\";\r\nimport { ChatMessageAttachment } from \"@banta/common\";\r\nimport { Subject } from \"rxjs\";\r\n\r\nlet TWITTER_LOADED = false;\r\n\r\n@Component({\r\n    selector: 'banta-attachment',\r\n    templateUrl: './attachment.component.html',\r\n    styleUrls: ['./attachment.component.scss']\r\n})\r\nexport class BantaAttachmentComponent {\r\n    constructor(\r\n        private elementRef: ElementRef<HTMLElement>\r\n    ) {\r\n\r\n    }\r\n\r\n    private _attachment: ChatMessageAttachment;\r\n    @Input() get attachment() { return this._attachment; }\r\n    set attachment(value) {\r\n        this._attachment = value;\r\n        this.checkLoad();\r\n        this.loadPlatformSpecific();\r\n    }\r\n    \r\n    @Input() loading = false;\r\n    @Input() editing = false;\r\n    @Input() loadingMessage: string = 'Please wait...';\r\n    @Input() error = false;\r\n    @Input() errorMessage: string = 'An error has occurred';\r\n    @Output() removed = new Subject<void>();\r\n    @Output() activated = new Subject<void>();\r\n    @Output() loaded = new Subject<void>();\r\n\r\n    private _viewLoaded = false;\r\n    ngAfterViewInit() {\r\n        this._viewLoaded = true;\r\n        this.checkLoad();\r\n    }\r\n\r\n    private loadTwitterWidgets() {\r\n        if (typeof window !== 'undefined')\r\n            window['twttr']?.widgets.load();\r\n    }\r\n\r\n    private loadPlatformSpecific() {\r\n        if (typeof window === 'undefined')\r\n            return;\r\n        \r\n        if (this._attachment?.type === 'tweet') {\r\n            if (!TWITTER_LOADED && document.querySelector('script[src=\"https://platform.twitter.com/widgets.js\"]'))\r\n                TWITTER_LOADED = true;\r\n\r\n            if (typeof window !== 'undefined' && !TWITTER_LOADED) {\r\n                TWITTER_LOADED = true;\r\n                let script = document.createElement('script');\r\n                script.src = 'https://platform.twitter.com/widgets.js';\r\n                script.async = true;\r\n                script.addEventListener('load', () => setTimeout(() => this.loadTwitterWidgets()));\r\n                document.body.appendChild(script);\r\n            } else {\r\n                setTimeout(() => this.loadTwitterWidgets());\r\n            }\r\n        }\r\n    }\r\n\r\n    private checkLoad() {\r\n        if (!this._attachment || !this._viewLoaded || !this.elementRef?.nativeElement)\r\n            return;\r\n\r\n        if (typeof window === 'undefined')\r\n            this.loaded.next();\r\n        else\r\n            setTimeout(() => this.loaded.next(), 250);\r\n    }\r\n\r\n    activate() {\r\n        this.activated.next();\r\n    }\r\n\r\n    remove() {\r\n        this.removed.next();\r\n    }\r\n\r\n    get isError() {\r\n        return this.error || this.attachment?.transientState?.error;\r\n    }\r\n\r\n    get theErrorMessage() {\r\n        return this.errorMessage || this.attachment?.transientState?.errorMessage;\r\n    }\r\n\r\n    @HostBinding('class.loading')\r\n    get isLoading() {\r\n        return this.editing && (\r\n            this.loading || !this.attachment || this.attachment.transientState?.loading \r\n            || !this.attachment.url\r\n        );\r\n    }\r\n\r\n    get isImageAttachment() {\r\n        return this.attachment?.type?.startsWith('image/');\r\n    }\r\n\r\n    get hasFrame() {\r\n        if (!this.attachment)\r\n            return false;\r\n\r\n        return this.attachment.type === 'iframe' || (\r\n            this.attachment.type === 'card' \r\n            && this.attachment.card?.player\r\n        );\r\n    }\r\n\r\n    get frameUrl() {\r\n        if (!this.attachment)\r\n            return undefined;\r\n        \r\n        if (this.attachment.type === 'iframe') {\r\n            return this.attachment.url;\r\n        } else if (this.attachment.type === 'card') {\r\n            return this.attachment.card.player;\r\n        }\r\n    }\r\n}","<button type=\"button\" (click)=\"remove()\" mat-mini-fab color=\"primary\" class=\"remove-button\" *ngIf=\"editing\">\r\n    <mat-icon>close</mat-icon>\r\n</button>\r\n\r\n<ng-container *ngIf=\"isError\">\r\n    <mat-icon class=\"error\">close</mat-icon>\r\n    <em class=\"error\">{{theErrorMessage}}</em>\r\n</ng-container>\r\n<ng-container *ngIf=\"!isError\">\r\n    <ng-container *ngIf=\"isLoading\">\r\n        <mat-spinner></mat-spinner>\r\n        <em>{{loadingMessage}}</em>\r\n    </ng-container>\r\n    <ng-container *ngIf=\"!isLoading && attachment\">\r\n        <iframe *ngIf=\"hasFrame\"\r\n            sandbox=\"allow-scripts allow-popups allow-same-origin allow-presentation\" \r\n            [src]=\"frameUrl | trustResourceUrl\"></iframe>\r\n        <a *ngIf=\"attachment.type === 'card'\" class=\"card-attachment\" [href]=\"attachment.url\" target=\"_blank\" [class.has-image]=\"attachment.card.image\">\r\n            <img \r\n                *ngIf=\"attachment.card.image\"\r\n                class=\"thumbnail\" \r\n                [src]=\"attachment.card.image\"\r\n                />\r\n            <div class=\"description\">\r\n                <h1>{{attachment.card.title}}</h1>\r\n                <div class=\"summary\">\r\n                    {{attachment.card.description}}\r\n                </div>\r\n                <cite>{{attachment.card.url}}</cite>\r\n            </div>\r\n        </a>\r\n        <a class=\"image-attachment\" *ngIf=\"isImageAttachment && attachment.url\" href=\"javascript:;\" (click)=\"activate()\">\r\n            <img [src]=\"attachment.url\" alt=\"Image Attachment\">\r\n        </a>\r\n        <blockquote *ngIf=\"attachment.type === 'tweet'\" \r\n            class=\"twitter-tweet\">\r\n                <p lang=\"en\" dir=\"ltr\"></p>\r\n                <a [href]=\"attachment.url\"></a>\r\n        </blockquote>\r\n    </ng-container>\r\n</ng-container>"]}
@@ -68,7 +68,8 @@ export class BantaMarkdownToHtmlPipe {
68
68
  });
69
69
  value = twemoji.parse(value, { base: this.emojiUrl });
70
70
  return this.sanitizer.bypassSecurityTrustHtml(purifier.sanitize(value, {
71
- FORBID_TAGS: ['h1', 'h2', 'h3', 'h4'],
71
+ FORBID_TAGS: ['h1', 'h2', 'h3', 'h4', 'style', 'link', 'script'],
72
+ FORBID_ATTR: ['style'],
72
73
  KEEP_CONTENT: true
73
74
  }));
74
75
  }
@@ -86,4 +87,4 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "17.3.9", ngImpor
86
87
  }, {
87
88
  type: Optional
88
89
  }] }] });
89
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"markdown-to-html.pipe.js","sourceRoot":"","sources":["../../../../../projects/sdk/src/lib/common/markdown-to-html.pipe.ts"],"names":[],"mappings":"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","sourcesContent":["import { Pipe, PipeTransform, Inject, Optional } from '@angular/core';\r\nimport * as marked from 'marked';\r\nimport createDOMPurify from 'dompurify';\r\nimport { DomSanitizer } from '@angular/platform-browser';\r\nimport twemoji from 'twemoji';\r\nimport { BANTA_SDK_OPTIONS, SdkOptions } from '../sdk-options';\r\n\r\nconst underline = {\r\n    name: 'underline',\r\n    level: 'inline', // Is this a block-level or inline-level tokenizer?\r\n    start(src) { return src.match(/\\+\\+/)?.index; }, // Hint to Marked.js to stop and check for a match\r\n    tokenizer(src, tokens) {\r\n        const rule = /^\\+\\+(.*?)\\+\\+/; // Regex for the complete token\r\n        const match = rule.exec(src);\r\n        if (match) {\r\n            return { // Token to generate\r\n                type: 'underline', // Should match \"name\" above\r\n                raw: match[0], // Text to consume from the source\r\n                text: this.lexer.inlineTokens(match[1].trim()), // Additional custom properties\r\n            };\r\n        }\r\n    },\r\n    renderer(token) {\r\n        return `<u>${this.parser.parseInline(token.text)}</u>`;\r\n    }\r\n};\r\n\r\nmarked.marked.use({\r\n    extensions: [underline]\r\n});\r\n\r\n@Pipe({\r\n    name: 'markdownToHtml'\r\n})\r\nexport class BantaMarkdownToHtmlPipe implements PipeTransform {\r\n    constructor(\r\n        private sanitizer: DomSanitizer,\r\n\r\n        @Inject(BANTA_SDK_OPTIONS) @Optional()\r\n        private sdkOptions: SdkOptions\r\n    ) {\r\n        this.renderer = new marked.Renderer({\r\n            headerPrefix: ''\r\n        });\r\n        const linkRenderer = this.renderer.link;\r\n        this.renderer.link = (href, title, text) => {\r\n            const html = linkRenderer.call(this.renderer, href, title, text);\r\n            return html.replace(/^<a /, '<a target=\"_blank\" rel=\"noopener noreferrer nofollow\" ');\r\n        };\r\n    }\r\n\r\n\tprivate get emojiUrl() {\r\n\t\treturn this.sdkOptions?.emojiUrl ?? 'https://cdn.jsdelivr.net/gh/twitter/twemoji@14.0.2/assets/';\r\n\t}\r\n\r\n    renderer: marked.Renderer;\r\n    transform(value: string) {\r\n        if (!value)\r\n            return '';\r\n\r\n        let purifier = createDOMPurify(window);\r\n        \r\n        // https://github.com/cure53/DOMPurify/blob/e1c19cf6407d782b666cb1d02a6af191f9cbc09e/demos/hooks-target-blank-demo.html\r\n        // Add a hook to make all links open a new window\r\n        purifier.addHook('afterSanitizeAttributes', function(node: HTMLElement & { target?: string }) {\r\n            // set all elements owning target to target=_blank\r\n            if ('target' in node) {\r\n                node.setAttribute('target','_blank');\r\n                // prevent https://www.owasp.org/index.php/Reverse_Tabnabbing\r\n                node.setAttribute('rel', 'noopener noreferrer nofollow');\r\n            }\r\n            // set non-HTML/MathML links to xlink:show=new\r\n            if (!node.hasAttribute('target')\r\n                && (node.hasAttribute('xlink:href')\r\n                    || node.hasAttribute('href'))) {\r\n                node.setAttribute('xlink:show', 'new');\r\n            }\r\n        });\r\n\r\n        value = marked.marked.parse(value, {\r\n            renderer: this.renderer\r\n        });\r\n\r\n        value = twemoji.parse(value, { base: this.emojiUrl });\r\n\r\n        return this.sanitizer.bypassSecurityTrustHtml(\r\n            purifier.sanitize(value,\r\n                {\r\n                    FORBID_TAGS: ['h1', 'h2', 'h3', 'h4'],\r\n                    KEEP_CONTENT: true\r\n                }\r\n            )\r\n        );\r\n    }\r\n}"]}
90
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"markdown-to-html.pipe.js","sourceRoot":"","sources":["../../../../../projects/sdk/src/lib/common/markdown-to-html.pipe.ts"],"names":[],"mappings":"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","sourcesContent":["import { Pipe, PipeTransform, Inject, Optional } from '@angular/core';\r\nimport * as marked from 'marked';\r\nimport createDOMPurify from 'dompurify';\r\nimport { DomSanitizer } from '@angular/platform-browser';\r\nimport twemoji from 'twemoji';\r\nimport { BANTA_SDK_OPTIONS, SdkOptions } from '../sdk-options';\r\n\r\nconst underline = {\r\n    name: 'underline',\r\n    level: 'inline', // Is this a block-level or inline-level tokenizer?\r\n    start(src) { return src.match(/\\+\\+/)?.index; }, // Hint to Marked.js to stop and check for a match\r\n    tokenizer(src, tokens) {\r\n        const rule = /^\\+\\+(.*?)\\+\\+/; // Regex for the complete token\r\n        const match = rule.exec(src);\r\n        if (match) {\r\n            return { // Token to generate\r\n                type: 'underline', // Should match \"name\" above\r\n                raw: match[0], // Text to consume from the source\r\n                text: this.lexer.inlineTokens(match[1].trim()), // Additional custom properties\r\n            };\r\n        }\r\n    },\r\n    renderer(token) {\r\n        return `<u>${this.parser.parseInline(token.text)}</u>`;\r\n    }\r\n};\r\n\r\nmarked.marked.use({\r\n    extensions: [underline]\r\n});\r\n\r\n@Pipe({\r\n    name: 'markdownToHtml'\r\n})\r\nexport class BantaMarkdownToHtmlPipe implements PipeTransform {\r\n    constructor(\r\n        private sanitizer: DomSanitizer,\r\n\r\n        @Inject(BANTA_SDK_OPTIONS) @Optional()\r\n        private sdkOptions: SdkOptions\r\n    ) {\r\n        this.renderer = new marked.Renderer({\r\n            headerPrefix: ''\r\n        });\r\n        const linkRenderer = this.renderer.link;\r\n        this.renderer.link = (href, title, text) => {\r\n            const html = linkRenderer.call(this.renderer, href, title, text);\r\n            return html.replace(/^<a /, '<a target=\"_blank\" rel=\"noopener noreferrer nofollow\" ');\r\n        };\r\n    }\r\n\r\n\tprivate get emojiUrl() {\r\n\t\treturn this.sdkOptions?.emojiUrl ?? 'https://cdn.jsdelivr.net/gh/twitter/twemoji@14.0.2/assets/';\r\n\t}\r\n\r\n    renderer: marked.Renderer;\r\n    transform(value: string) {\r\n        if (!value)\r\n            return '';\r\n\r\n        let purifier = createDOMPurify(window);\r\n        \r\n        // https://github.com/cure53/DOMPurify/blob/e1c19cf6407d782b666cb1d02a6af191f9cbc09e/demos/hooks-target-blank-demo.html\r\n        // Add a hook to make all links open a new window\r\n        purifier.addHook('afterSanitizeAttributes', function(node: HTMLElement & { target?: string }) {\r\n            // set all elements owning target to target=_blank\r\n            if ('target' in node) {\r\n                node.setAttribute('target','_blank');\r\n                // prevent https://www.owasp.org/index.php/Reverse_Tabnabbing\r\n                node.setAttribute('rel', 'noopener noreferrer nofollow');\r\n            }\r\n            // set non-HTML/MathML links to xlink:show=new\r\n            if (!node.hasAttribute('target')\r\n                && (node.hasAttribute('xlink:href')\r\n                    || node.hasAttribute('href'))) {\r\n                node.setAttribute('xlink:show', 'new');\r\n            }\r\n        });\r\n\r\n        value = marked.marked.parse(value, {\r\n            renderer: this.renderer\r\n        });\r\n\r\n        value = twemoji.parse(value, { base: this.emojiUrl });\r\n\r\n        return this.sanitizer.bypassSecurityTrustHtml(\r\n            purifier.sanitize(value,\r\n                {\r\n                    FORBID_TAGS: ['h1', 'h2', 'h3', 'h4', 'style', 'link', 'script'],\r\n                    FORBID_ATTR: ['style'],\r\n                    KEEP_CONTENT: true\r\n                }\r\n            )\r\n        );\r\n    }\r\n}"]}
@@ -349,7 +349,8 @@ class BantaMarkdownToHtmlPipe {
349
349
  });
350
350
  value = twemoji$1.parse(value, { base: this.emojiUrl });
351
351
  return this.sanitizer.bypassSecurityTrustHtml(purifier.sanitize(value, {
352
- FORBID_TAGS: ['h1', 'h2', 'h3', 'h4'],
352
+ FORBID_TAGS: ['h1', 'h2', 'h3', 'h4', 'style', 'link', 'script'],
353
+ FORBID_ATTR: ['style'],
353
354
  KEEP_CONTENT: true
354
355
  }));
355
356
  }
@@ -449,6 +450,8 @@ class BantaAttachmentComponent {
449
450
  window['twttr']?.widgets.load();
450
451
  }
451
452
  loadPlatformSpecific() {
453
+ if (typeof window === 'undefined')
454
+ return;
452
455
  if (this._attachment?.type === 'tweet') {
453
456
  if (!TWITTER_LOADED && document.querySelector('script[src="https://platform.twitter.com/widgets.js"]'))
454
457
  TWITTER_LOADED = true;
@@ -9365,7 +9368,7 @@ class ChatMessageComponent {
9365
9368
  this._selected.next();
9366
9369
  }
9367
9370
  selectUser() {
9368
- this._userSelected.next();
9371
+ this._userSelected.next(this.message.user);
9369
9372
  }
9370
9373
  static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "17.3.9", ngImport: i0, type: ChatMessageComponent, deps: [], target: i0.ɵɵFactoryTarget.Component }); }
9371
9374
  static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "17.3.9", type: ChatMessageComponent, selector: "banta-chat-message", inputs: { message: "message" }, outputs: { userSelected: "userSelected", selected: "selected", reported: "reported", upvoted: "upvoted" }, ngImport: i0, template: "<div class=\"message-content\">\r\n <div class=\"user\" (click)=\"selectUser()\">\r\n <div class=\"avatar\" [style.background-image]=\"avatarForUser(message.user)\"></div>\r\n <label>{{message.user.username}}</label>\r\n </div>\r\n <div class=\"content\">\r\n <div (click)=\"select()\">\r\n {{message.message}}\r\n </div>\r\n <div class=\"status\">\r\n <div class=\"count-indicator\" *ngIf=\"message.likes > 0\">\r\n <mat-icon>star</mat-icon> {{message.likes}}\r\n </div>\r\n </div>\r\n </div>\r\n</div>\r\n<div class=\"actions\">\r\n <button mat-icon-button matTooltip=\"Upvote\" matTooltipPosition=\"below\" (click)=\"upvote()\">\r\n <mat-icon>thumb_up</mat-icon>\r\n </button>\r\n <button mat-icon-button matTooltip=\"Report\" matTooltipPosition=\"below\" (click)=\"report()\">\r\n <mat-icon>report</mat-icon>\r\n </button>\r\n</div>", styles: [":host{display:flex;flex-direction:row;align-items:center;padding:0 1em;background-color:#fff;color:#000;transition:.4s background-color ease-out}:host .message-content .content{color:#111}:host:hover{background-color:#ddd}:host.highlight{background:#00121b}:host.highlight:hover{background:#01324d}:host:nth-child(2n){background-color:#eee}:host:nth-child(2n):hover{background:#ddd}:host:nth-child(2n) .message-content .content{color:#222}:host:nth-child(2n).highlight{background:#001a2a}:host:nth-child(2n).highlight:hover{background:#002b44}:host .message-content{display:flex;flex-direction:row;flex-grow:1;align-items:center}:host .message-content .content{display:flex;flex-direction:row;padding:5px 0}:host .message-content .content .status{display:flex;flex-direction:row;align-items:center;margin-left:1em}:host .message-content .content .status mat-icon{margin-left:.5em}:host .user{color:#999;font-weight:400;text-align:right;margin-right:.25em;flex-shrink:0;display:flex;align-items:center}:host .user .avatar{background-position:center;background-size:cover;background-color:#333;border-radius:100%;flex-shrink:0;flex-grow:0;margin-right:1em;width:2em;height:2em}:host .user:after{content:\":\";margin-right:1em}:host .content{flex-grow:1}:host .actions{flex-shrink:0;white-space:nowrap;opacity:0;transition:.4s opacity ease-out}:host:hover .actions{opacity:1}.count-indicator{white-space:nowrap}.count-indicator mat-icon{vertical-align:bottom}:host-context(.mat-dark-theme){background-color:#000;color:#fff}:host-context(.mat-dark-theme) .message-content .content{color:#ddd}:host-context(.mat-dark-theme):hover{background-color:#111}:host-context(.mat-dark-theme):nth-child(2n).highlight{background:#001a2a}:host-context(.mat-dark-theme):nth-child(2n).highlight:hover{background:#002b44}:host-context(.mat-dark-theme):nth-child(2n):hover{background-color:#111}:host-context(.mat-dark-theme):nth-child(2n){background-color:#080808}:host-context(.mat-dark-theme):nth-child(2n) .message-content .content{color:#eee}label{margin:0}\n"], dependencies: [{ kind: "directive", type: i2.NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }, { kind: "component", type: i2$1.MatIcon, selector: "mat-icon", inputs: ["color", "inline", "svgIcon", "fontSet", "fontIcon"], exportAs: ["matIcon"] }, { kind: "component", type: i6.MatIconButton, selector: "button[mat-icon-button]", exportAs: ["matButton"] }] }); }