mirror of
https://github.com/Alfresco/alfresco-ng2-components.git
synced 2025-05-26 17:24:56 +00:00
Merge branch 'development' of https://github.com/Alfresco/alfresco-ng2-components into development
This commit is contained in:
commit
7ad77493d7
@ -77,7 +77,7 @@
|
|||||||
key="archivedAt"
|
key="archivedAt"
|
||||||
title="DOCUMENT_LIST.COLUMNS.DELETED_ON">
|
title="DOCUMENT_LIST.COLUMNS.DELETED_ON">
|
||||||
<ng-template let-value="value">
|
<ng-template let-value="value">
|
||||||
<span title="{{ value | date:'medium' }}">{{ value | adfTimeAgo }}</span>
|
<span title="{{ value | date:'medium' }}">{{ value | adfTimeAgo: currentLocale }}</span>
|
||||||
</ng-template>
|
</ng-template>
|
||||||
</data-column>
|
</data-column>
|
||||||
|
|
||||||
|
@ -37,12 +37,17 @@ export class TrashcanComponent {
|
|||||||
documentList: DocumentListComponent;
|
documentList: DocumentListComponent;
|
||||||
|
|
||||||
supportedPages = [];
|
supportedPages = [];
|
||||||
|
currentLocale;
|
||||||
|
|
||||||
constructor(private preference: UserPreferencesService) {
|
constructor(private preference: UserPreferencesService) {
|
||||||
this.preference.select(UserPreferenceValues.SupportedPageSizes)
|
this.preference.select(UserPreferenceValues.SupportedPageSizes)
|
||||||
.subscribe((pages) => {
|
.subscribe((pages) => {
|
||||||
this.supportedPages = pages;
|
this.supportedPages = pages;
|
||||||
});
|
});
|
||||||
|
|
||||||
|
this.preference.select(UserPreferenceValues.Locale).subscribe((locale) => {
|
||||||
|
this.currentLocale = locale;
|
||||||
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
refresh() {
|
refresh() {
|
||||||
|
@ -39,6 +39,7 @@ Activates a file upload.
|
|||||||
| tooltip | `string` | null | Custom tooltip text. |
|
| tooltip | `string` | null | Custom tooltip text. |
|
||||||
| uploadFolders | `boolean` | false | Allows/disallows upload folders (only for Chrome). |
|
| uploadFolders | `boolean` | false | Allows/disallows upload folders (only for Chrome). |
|
||||||
| versioning | `boolean` | false | Toggles versioning. |
|
| versioning | `boolean` | false | Toggles versioning. |
|
||||||
|
| nodeType | `string` | "cm:content" | Custom node type for uploaded file. |
|
||||||
|
|
||||||
### Events
|
### Events
|
||||||
|
|
||||||
|
@ -108,46 +108,6 @@ describe('ShareDataTableAdapter', () => {
|
|||||||
expect(check).toThrowError(adapter.ERR_COL_NOT_FOUND);
|
expect(check).toThrowError(adapter.ERR_COL_NOT_FOUND);
|
||||||
});
|
});
|
||||||
|
|
||||||
it('should covert cell value to formatted date', () => {
|
|
||||||
let rawValue = new Date(2015, 6, 15, 21, 43, 11); // Wed Jul 15 2015 21:43:11 GMT+0100 (BST);
|
|
||||||
let dateValue = 'Jul 15, 2015, 9:43:11 PM';
|
|
||||||
|
|
||||||
let file = new FileNode();
|
|
||||||
file.entry.createdAt = rawValue;
|
|
||||||
|
|
||||||
let col = <DataColumn> {
|
|
||||||
key: 'createdAt',
|
|
||||||
type: 'date',
|
|
||||||
format: 'medium' // Jul 15, 2015, 9:43:11 PM
|
|
||||||
};
|
|
||||||
|
|
||||||
let row = new ShareDataRow(file, documentListService, null);
|
|
||||||
let adapter = new ShareDataTableAdapter(documentListService, null);
|
|
||||||
|
|
||||||
let value = adapter.getValue(row, col);
|
|
||||||
expect(value).toBe(dateValue);
|
|
||||||
});
|
|
||||||
|
|
||||||
it('should use default date format as fallback', () => {
|
|
||||||
let rawValue = new Date(2015, 6, 15, 21, 43, 11); // Wed Jul 15 2015 21:43:11 GMT+0100 (BST);
|
|
||||||
let dateValue = 'Jul 15, 2015, 9:43:11 PM';
|
|
||||||
|
|
||||||
let file = new FileNode();
|
|
||||||
file.entry.createdAt = rawValue;
|
|
||||||
|
|
||||||
let col = <DataColumn> {
|
|
||||||
key: 'createdAt',
|
|
||||||
type: 'date',
|
|
||||||
format: null
|
|
||||||
};
|
|
||||||
|
|
||||||
let row = new ShareDataRow(file, documentListService, null);
|
|
||||||
let adapter = new ShareDataTableAdapter(documentListService, null);
|
|
||||||
|
|
||||||
let value = adapter.getValue(row, col);
|
|
||||||
expect(value).toBe(dateValue);
|
|
||||||
});
|
|
||||||
|
|
||||||
it('should return date value as string', () => {
|
it('should return date value as string', () => {
|
||||||
let rawValue = new Date(2015, 6, 15, 21, 43, 11); // Wed Jul 15 2015 21:43:11 GMT+0100 (BST);
|
let rawValue = new Date(2015, 6, 15, 21, 43, 11); // Wed Jul 15 2015 21:43:11 GMT+0100 (BST);
|
||||||
|
|
||||||
@ -166,26 +126,6 @@ describe('ShareDataTableAdapter', () => {
|
|||||||
expect(value).toBe(rawValue);
|
expect(value).toBe(rawValue);
|
||||||
});
|
});
|
||||||
|
|
||||||
it('should log error when having date conversion issues', () => {
|
|
||||||
let dateValue = <Date> {};
|
|
||||||
let file = new FileNode();
|
|
||||||
file.entry.createdAt = <any> dateValue;
|
|
||||||
|
|
||||||
let col = <DataColumn> {
|
|
||||||
key: 'createdAt',
|
|
||||||
type: 'date',
|
|
||||||
format: 'medium'
|
|
||||||
};
|
|
||||||
|
|
||||||
let row = new ShareDataRow(file, documentListService, null);
|
|
||||||
let adapter = new ShareDataTableAdapter(documentListService, null);
|
|
||||||
spyOn(console, 'error').and.stub();
|
|
||||||
|
|
||||||
let value = adapter.getValue(row, col);
|
|
||||||
expect(value).toBe('Error');
|
|
||||||
expect(console.error).toHaveBeenCalled();
|
|
||||||
});
|
|
||||||
|
|
||||||
it('should generate fallback icon for a file thumbnail with missing mime type', () => {
|
it('should generate fallback icon for a file thumbnail with missing mime type', () => {
|
||||||
spyOn(documentListService, 'getDefaultMimeTypeIcon').and.returnValue(`assets/images/ft_ic_miscellaneouse.svg`);
|
spyOn(documentListService, 'getDefaultMimeTypeIcon').and.returnValue(`assets/images/ft_ic_miscellaneouse.svg`);
|
||||||
|
|
||||||
|
@ -16,8 +16,6 @@
|
|||||||
*/
|
*/
|
||||||
|
|
||||||
import { DataColumn, DataRow, DataSorting, DataTableAdapter } from '@alfresco/adf-core';
|
import { DataColumn, DataRow, DataSorting, DataTableAdapter } from '@alfresco/adf-core';
|
||||||
import { TimeAgoPipe } from '@alfresco/adf-core';
|
|
||||||
import { DatePipe } from '@angular/common';
|
|
||||||
import { NodePaging } from 'alfresco-js-api';
|
import { NodePaging } from 'alfresco-js-api';
|
||||||
import { PermissionStyleModel } from './../models/permissions-style.model';
|
import { PermissionStyleModel } from './../models/permissions-style.model';
|
||||||
import { DocumentListService } from './../services/document-list.service';
|
import { DocumentListService } from './../services/document-list.service';
|
||||||
@ -78,16 +76,6 @@ export class ShareDataTableAdapter implements DataTableAdapter {
|
|||||||
return dataRow.cache[col.key];
|
return dataRow.cache[col.key];
|
||||||
}
|
}
|
||||||
|
|
||||||
if (col.type === 'date') {
|
|
||||||
try {
|
|
||||||
const result = this.formatDate(col, value);
|
|
||||||
return dataRow.cacheValue(col.key, result);
|
|
||||||
} catch (err) {
|
|
||||||
console.error(`Error parsing date ${value} to format ${col.format}`);
|
|
||||||
return 'Error';
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
if (col.key === '$thumbnail') {
|
if (col.key === '$thumbnail') {
|
||||||
|
|
||||||
if (this.imageResolver) {
|
if (this.imageResolver) {
|
||||||
@ -132,21 +120,6 @@ export class ShareDataTableAdapter implements DataTableAdapter {
|
|||||||
return dataRow.cacheValue(col.key, value);
|
return dataRow.cacheValue(col.key, value);
|
||||||
}
|
}
|
||||||
|
|
||||||
formatDate(col: DataColumn, value: any): string {
|
|
||||||
if (col.type === 'date') {
|
|
||||||
const format = col.format || 'medium';
|
|
||||||
if (format === 'timeAgo') {
|
|
||||||
const timeAgoPipe = new TimeAgoPipe();
|
|
||||||
return timeAgoPipe.transform(value);
|
|
||||||
} else {
|
|
||||||
const datePipe = new DatePipe('en-US');
|
|
||||||
return datePipe.transform(value, format);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
return value;
|
|
||||||
}
|
|
||||||
|
|
||||||
getSorting(): DataSorting {
|
getSorting(): DataSorting {
|
||||||
return this.sorting;
|
return this.sorting;
|
||||||
}
|
}
|
||||||
|
@ -24,7 +24,7 @@ import { NoopAnimationsModule } from '@angular/platform-browser/animations';
|
|||||||
|
|
||||||
@Component({
|
@Component({
|
||||||
selector: 'adf-upload-button-test',
|
selector: 'adf-upload-button-test',
|
||||||
template: 'test componente';
|
template: 'test componente'
|
||||||
})
|
})
|
||||||
|
|
||||||
export class UploadTestComponent extends UploadBase {
|
export class UploadTestComponent extends UploadBase {
|
||||||
@ -214,11 +214,12 @@ describe('UploadBase', () => {
|
|||||||
component.uploadFiles(files);
|
component.uploadFiles(files);
|
||||||
|
|
||||||
expect(addToQueueSpy).toHaveBeenCalledWith(new FileModel(files[0], {
|
expect(addToQueueSpy).toHaveBeenCalledWith(new FileModel(files[0], {
|
||||||
comment: 'example-comment'
|
comment: 'example-comment',
|
||||||
newVersion: false,
|
newVersion: false,
|
||||||
majorVersion: false,
|
majorVersion: false,
|
||||||
parentId: '-root-',
|
parentId: '-root-',
|
||||||
path: ''
|
path: '',
|
||||||
|
nodeType: 'cm:content'
|
||||||
}));
|
}));
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
@ -246,7 +247,8 @@ describe('UploadBase', () => {
|
|||||||
newVersion: true,
|
newVersion: true,
|
||||||
majorVersion: true,
|
majorVersion: true,
|
||||||
parentId: '-root-',
|
parentId: '-root-',
|
||||||
path: ''
|
path: '',
|
||||||
|
nodeType: 'cm:content'
|
||||||
}));
|
}));
|
||||||
});
|
});
|
||||||
|
|
||||||
@ -261,7 +263,49 @@ describe('UploadBase', () => {
|
|||||||
newVersion: true,
|
newVersion: true,
|
||||||
majorVersion: false,
|
majorVersion: false,
|
||||||
parentId: '-root-',
|
parentId: '-root-',
|
||||||
path: ''
|
path: '',
|
||||||
|
nodeType: 'cm:content'
|
||||||
|
}));
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
describe('Node Type', () => {
|
||||||
|
|
||||||
|
let addToQueueSpy;
|
||||||
|
|
||||||
|
const files: File[] = [
|
||||||
|
<File> { name: 'process.pbmn' }
|
||||||
|
];
|
||||||
|
|
||||||
|
beforeEach(() => {
|
||||||
|
addToQueueSpy = spyOn(uploadService, 'addToQueue');
|
||||||
|
});
|
||||||
|
|
||||||
|
it('should have custom nodeType if it is set', () => {
|
||||||
|
component.nodeType = 'ama:process';
|
||||||
|
|
||||||
|
component.uploadFiles(files);
|
||||||
|
|
||||||
|
expect(addToQueueSpy).toHaveBeenCalledWith(new FileModel(files[0], {
|
||||||
|
comment: undefined,
|
||||||
|
newVersion: false,
|
||||||
|
majorVersion: false,
|
||||||
|
parentId: '-root-',
|
||||||
|
path: '',
|
||||||
|
nodeType: 'ama:process'
|
||||||
|
}));
|
||||||
|
});
|
||||||
|
|
||||||
|
it('should have default nodeType if it is not set', () => {
|
||||||
|
component.uploadFiles(files);
|
||||||
|
|
||||||
|
expect(addToQueueSpy).toHaveBeenCalledWith(new FileModel(files[0], {
|
||||||
|
comment: undefined,
|
||||||
|
newVersion: false,
|
||||||
|
majorVersion: false,
|
||||||
|
parentId: '-root-',
|
||||||
|
path: '',
|
||||||
|
nodeType: 'cm:content'
|
||||||
}));
|
}));
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
|
@ -53,6 +53,10 @@ export abstract class UploadBase {
|
|||||||
@Input()
|
@Input()
|
||||||
comment: string;
|
comment: string;
|
||||||
|
|
||||||
|
/** Custom node type for uploaded file */
|
||||||
|
@Input()
|
||||||
|
nodeType: string = 'cm:content';
|
||||||
|
|
||||||
/** Emitted when the file is uploaded successfully. */
|
/** Emitted when the file is uploaded successfully. */
|
||||||
@Output()
|
@Output()
|
||||||
success = new EventEmitter();
|
success = new EventEmitter();
|
||||||
@ -139,7 +143,8 @@ export abstract class UploadBase {
|
|||||||
majorVersion: this.majorVersion,
|
majorVersion: this.majorVersion,
|
||||||
newVersion: this.versioning,
|
newVersion: this.versioning,
|
||||||
parentId: parentId,
|
parentId: parentId,
|
||||||
path: path
|
path: path,
|
||||||
|
nodeType: this.nodeType
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -107,7 +107,7 @@
|
|||||||
<mat-icon>{{ data.getValue(row, col) }}</mat-icon>
|
<mat-icon>{{ data.getValue(row, col) }}</mat-icon>
|
||||||
</div>
|
</div>
|
||||||
<div *ngSwitchCase="'date'" class="cell-value"
|
<div *ngSwitchCase="'date'" class="cell-value"
|
||||||
[attr.data-automation-id]="'date_' + data.getValue(row, col)">
|
[attr.data-automation-id]="'date_' + (data.getValue(row, col) | date: 'medium') ">
|
||||||
<adf-date-cell
|
<adf-date-cell
|
||||||
[data]="data"
|
[data]="data"
|
||||||
[column]="col"
|
[column]="col"
|
||||||
|
@ -15,18 +15,38 @@
|
|||||||
* limitations under the License.
|
* limitations under the License.
|
||||||
*/
|
*/
|
||||||
|
|
||||||
import { ChangeDetectionStrategy, Component, ViewEncapsulation } from '@angular/core';
|
import { Component, ViewEncapsulation } from '@angular/core';
|
||||||
import { DataTableCellComponent } from './datatable-cell.component';
|
import { DataTableCellComponent } from './datatable-cell.component';
|
||||||
|
import { UserPreferencesService, UserPreferenceValues } from '../../../services';
|
||||||
|
|
||||||
@Component({
|
@Component({
|
||||||
selector: 'adf-date-cell',
|
selector: 'adf-date-cell',
|
||||||
changeDetection: ChangeDetectionStrategy.OnPush,
|
|
||||||
template: `
|
template: `
|
||||||
<ng-container>
|
<ng-container>
|
||||||
<span [title]="tooltip">{{value}}</span>
|
<span title="{{ tooltip | date:'medium' }}" *ngIf="column?.format === 'timeAgo' else standard_date">
|
||||||
|
{{ value | adfTimeAgo: currentLocale }}
|
||||||
|
</span>
|
||||||
</ng-container>
|
</ng-container>
|
||||||
|
<ng-template #standard_date>
|
||||||
|
<span title="{{ tooltip | date:'medium' }}">
|
||||||
|
{{ value | date:'medium' }}
|
||||||
|
</span>
|
||||||
|
</ng-template>
|
||||||
`,
|
`,
|
||||||
encapsulation: ViewEncapsulation.None,
|
encapsulation: ViewEncapsulation.None,
|
||||||
host: { class: 'adf-date-cell' }
|
host: { class: 'adf-date-cell' }
|
||||||
})
|
})
|
||||||
export class DateCellComponent extends DataTableCellComponent {}
|
export class DateCellComponent extends DataTableCellComponent {
|
||||||
|
|
||||||
|
currentLocale;
|
||||||
|
|
||||||
|
constructor(userPreferenceService: UserPreferencesService) {
|
||||||
|
super();
|
||||||
|
|
||||||
|
userPreferenceService.select(UserPreferenceValues.Locale).subscribe((locale) => {
|
||||||
|
this.currentLocale = locale;
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
||||||
|
@ -15,9 +15,6 @@
|
|||||||
* limitations under the License.
|
* limitations under the License.
|
||||||
*/
|
*/
|
||||||
|
|
||||||
import { DatePipe } from '@angular/common';
|
|
||||||
|
|
||||||
import { TimeAgoPipe } from '../../pipes';
|
|
||||||
import { DataColumn } from './data-column.model';
|
import { DataColumn } from './data-column.model';
|
||||||
import { DataRow } from './data-row.model';
|
import { DataRow } from './data-row.model';
|
||||||
import { ObjectDataRow } from './object-datarow.model';
|
import { ObjectDataRow } from './object-datarow.model';
|
||||||
@ -107,14 +104,6 @@ export class ObjectDataTableAdapter implements DataTableAdapter {
|
|||||||
|
|
||||||
let value = row.getValue(col.key);
|
let value = row.getValue(col.key);
|
||||||
|
|
||||||
if (col.type === 'date') {
|
|
||||||
try {
|
|
||||||
return this.formatDate(col, value);
|
|
||||||
} catch (err) {
|
|
||||||
console.error(`Error parsing date ${value} to format ${col.format}`);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
if (col.type === 'icon') {
|
if (col.type === 'icon') {
|
||||||
const icon = row.getValue(col.key);
|
const icon = row.getValue(col.key);
|
||||||
return icon;
|
return icon;
|
||||||
@ -123,21 +112,6 @@ export class ObjectDataTableAdapter implements DataTableAdapter {
|
|||||||
return value;
|
return value;
|
||||||
}
|
}
|
||||||
|
|
||||||
formatDate(col: DataColumn, value: any): string {
|
|
||||||
if (col.type === 'date') {
|
|
||||||
const format = col.format || 'medium';
|
|
||||||
if (format === 'timeAgo') {
|
|
||||||
const timeAgoPipe = new TimeAgoPipe();
|
|
||||||
return timeAgoPipe.transform(value);
|
|
||||||
} else {
|
|
||||||
const datePipe = new DatePipe('en-US');
|
|
||||||
return datePipe.transform(value, format);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
return value;
|
|
||||||
}
|
|
||||||
|
|
||||||
getSorting(): DataSorting {
|
getSorting(): DataSorting {
|
||||||
return this._sorting;
|
return this._sorting;
|
||||||
}
|
}
|
||||||
|
@ -28,6 +28,7 @@ export interface FileUploadOptions {
|
|||||||
newVersionBaseName?: string;
|
newVersionBaseName?: string;
|
||||||
parentId?: string;
|
parentId?: string;
|
||||||
path?: string;
|
path?: string;
|
||||||
|
nodeType?: string;
|
||||||
}
|
}
|
||||||
|
|
||||||
export enum FileUploadStatus {
|
export enum FileUploadStatus {
|
||||||
|
@ -16,14 +16,15 @@
|
|||||||
*/
|
*/
|
||||||
|
|
||||||
import { TimeAgoPipe } from './time-ago.pipe';
|
import { TimeAgoPipe } from './time-ago.pipe';
|
||||||
|
import { async } from '@angular/core/testing';
|
||||||
|
|
||||||
describe('TimeAgoPipe', () => {
|
describe('TimeAgoPipe', () => {
|
||||||
|
|
||||||
let pipe: TimeAgoPipe;
|
let pipe: TimeAgoPipe;
|
||||||
|
|
||||||
beforeEach(() => {
|
beforeEach(async(() => {
|
||||||
pipe = new TimeAgoPipe();
|
pipe = new TimeAgoPipe();
|
||||||
});
|
}));
|
||||||
|
|
||||||
it('should return time difference for a given date', () => {
|
it('should return time difference for a given date', () => {
|
||||||
let date = new Date();
|
let date = new Date();
|
||||||
@ -39,4 +40,13 @@ describe('TimeAgoPipe', () => {
|
|||||||
expect(pipe.transform(null)).toBe('');
|
expect(pipe.transform(null)).toBe('');
|
||||||
expect(pipe.transform(undefined)).toBe('');
|
expect(pipe.transform(undefined)).toBe('');
|
||||||
});
|
});
|
||||||
|
|
||||||
|
describe('When a locale is given', () => {
|
||||||
|
|
||||||
|
it('should return a localised message', async(() => {
|
||||||
|
let date = new Date();
|
||||||
|
const transformedDate = pipe.transform(date, 'de');
|
||||||
|
expect(transformedDate).toBe('vor ein paar Sekunden');
|
||||||
|
}));
|
||||||
|
});
|
||||||
});
|
});
|
||||||
|
@ -16,7 +16,6 @@
|
|||||||
*/
|
*/
|
||||||
|
|
||||||
import moment from 'moment-es6';
|
import moment from 'moment-es6';
|
||||||
|
|
||||||
import { Pipe, PipeTransform } from '@angular/core';
|
import { Pipe, PipeTransform } from '@angular/core';
|
||||||
|
|
||||||
@Pipe({
|
@Pipe({
|
||||||
@ -24,11 +23,14 @@ import { Pipe, PipeTransform } from '@angular/core';
|
|||||||
})
|
})
|
||||||
export class TimeAgoPipe implements PipeTransform {
|
export class TimeAgoPipe implements PipeTransform {
|
||||||
|
|
||||||
transform(value: Date) {
|
defaultLocale = 'en-US';
|
||||||
|
|
||||||
|
transform(value: Date, locale?: string) {
|
||||||
if (value !== null && value !== undefined ) {
|
if (value !== null && value !== undefined ) {
|
||||||
|
const actualLocale = locale ? locale : this.defaultLocale;
|
||||||
const then = moment(value);
|
const then = moment(value);
|
||||||
const diff = moment().diff(then, 'days');
|
const diff = moment().locale(actualLocale).diff(then, 'days');
|
||||||
return diff > 7 ? then.format('DD/MM/YYYY HH:mm') : then.fromNow();
|
return diff > 7 ? then.locale(actualLocale).format('DD/MM/YYYY HH:mm') : then.locale(actualLocale).fromNow();
|
||||||
}
|
}
|
||||||
return '';
|
return '';
|
||||||
}
|
}
|
||||||
|
@ -172,6 +172,10 @@ export class UploadService {
|
|||||||
opts.name = file.options.newVersionBaseName;
|
opts.name = file.options.newVersionBaseName;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if (file.options.nodeType) {
|
||||||
|
opts.nodeType = file.options.nodeType;
|
||||||
|
}
|
||||||
|
|
||||||
return this.apiService.getInstance().upload.uploadFile(
|
return this.apiService.getInstance().upload.uploadFile(
|
||||||
file.file,
|
file.file,
|
||||||
file.options.path,
|
file.options.path,
|
||||||
|
20789
package-lock.json
generated
20789
package-lock.json
generated
File diff suppressed because it is too large
Load Diff
Loading…
x
Reference in New Issue
Block a user