Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Content reports ported from DSpace 6.x #2163

Merged
merged 46 commits into from
Feb 28, 2024
Merged
Show file tree
Hide file tree
Changes from 11 commits
Commits
Show all changes
46 commits
Select commit Hold shift + click to select a range
f37122b
Content Reports
Nov 30, 2022
1d1c713
Fixed merge conflict
Dec 15, 2022
be97cad
Resolved conflict on pl.json5
Jan 9, 2023
bc73039
Merge branch 'main' of github.com:jeffmorin/dspace-angular
Jan 12, 2023
9533713
Synchronized API update with REST layer + i18n update in Greek and Uk…
Jan 12, 2023
9677de7
Fixed code style errors
Jan 12, 2023
97c6bf5
Merge branch 'DSpace:main' into main
jeffmorin Jan 12, 2023
18f913e
Merge branch 'DSpace:main' into main
jeffmorin Jan 16, 2023
2b8c377
Merge branch 'DSpace:main' into main
jeffmorin Jan 18, 2023
a105dd5
Fixed conflicts
Mar 24, 2023
2c0e0ab
Fixed conflicts and added strings for Content Reports
Mar 24, 2023
5a01670
Merge branch 'DSpace:main' into main
jeffmorin Apr 5, 2023
75ca560
Merge branch 'DSpace:main' into main
jeffmorin Apr 12, 2023
1cdbfc5
Resolved conflicts
Apr 20, 2023
0e2b767
Merge branch 'DSpace:main' into main
jeffmorin May 1, 2023
de83b35
Merge branch 'DSpace:main' into main
jeffmorin May 25, 2023
6d9768e
Updated to latest version from main branch
Nov 21, 2023
ef651a3
Resolved conflicts
Dec 18, 2023
16708d6
Merge branch 'DSpace:main' into main
jeffmorin Feb 12, 2024
d686cb1
Merge branch 'DSpace:main' into main
jeffmorin Feb 15, 2024
5f88a07
Removed non-working tests
Feb 15, 2024
4cd20d6
Removed unneeded actions file and fixed invalid TS file
Feb 15, 2024
85630ba
Fixed i18n files
Feb 15, 2024
872a670
Merge branch 'DSpace:main' into main
jeffmorin Feb 16, 2024
cadd1df
Applied Bootstrap styles to tables and form controls
Feb 19, 2024
998dd94
Merge branch 'main' of github.com:jeffmorin/dspace-angular
Feb 19, 2024
2d2a74a
Merge branch 'DSpace:main' into main
jeffmorin Feb 20, 2024
8bcd82e
Forgot styling on two buttons
Feb 21, 2024
27a9ce5
Merge branch 'main' of github.com:jeffmorin/dspace-angular
Feb 21, 2024
2a65bd8
Merge branch 'DSpace:main' into main
jeffmorin Feb 21, 2024
79993ce
Merge branch 'DSpace:main' into main
jeffmorin Feb 22, 2024
513b28b
Merge branch 'DSpace:main' into main
jeffmorin Feb 22, 2024
64e3149
Added enable/disable Conte Reports functionality
Feb 22, 2024
f1cfe99
Fixed single-quote constraint
Feb 22, 2024
5f41bc2
The newly added ConfigurationDataService seemed to be missing in the …
Feb 22, 2024
1c6216d
Revert to original version
Feb 22, 2024
c764950
Second attempt to fix menu.resolver.spec.ts
Feb 22, 2024
0497991
Merge branch 'DSpace:main' into main
jeffmorin Feb 23, 2024
e2bf9d2
Switched to GET requests for Content Reports
Feb 23, 2024
1cea469
Fixed styling errors
Feb 23, 2024
600c991
Fixed for loops over arrays
Feb 23, 2024
91d97f8
Fixed unit test
Feb 23, 2024
ba5f50e
Merge branch 'DSpace:main' into main
jeffmorin Feb 27, 2024
92553e0
Requested changes and fixes
Feb 27, 2024
26caab1
Merge branch 'main' of github.com:jeffmorin/dspace-angular
Feb 27, 2024
410a278
Resolved conflicts
Feb 28, 2024
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
37 changes: 37 additions & 0 deletions src/app/admin/admin-reports/admin-reports-routing.module.ts
Original file line number Diff line number Diff line change
@@ -0,0 +1,37 @@
import { FilteredCollectionsComponent } from './filtered-collections/filtered-collections.component';
import { FilteredItemsComponent } from './filtered-items/filtered-items.component';
import { RouterModule } from '@angular/router';
import { NgModule } from '@angular/core';
import { I18nBreadcrumbResolver } from '../../core/breadcrumbs/i18n-breadcrumb.resolver';

@NgModule({
imports: [
RouterModule.forChild([
{
path: 'collections',
resolve: { breadcrumb: I18nBreadcrumbResolver },
data: {title: 'admin.reports.collections.title', breadcrumbKey: 'admin.reports.collections'},
children: [
{
path: '',
component: FilteredCollectionsComponent
}
]
},
{
path: 'queries',
resolve: { breadcrumb: I18nBreadcrumbResolver },
data: {title: 'admin.reports.items.title', breadcrumbKey: 'admin.reports.items'},
children: [
{
path: '',
component: FilteredItemsComponent
}
]
}
])
]
})
export class AdminReportsRoutingModule {

}
28 changes: 28 additions & 0 deletions src/app/admin/admin-reports/admin-reports.module.ts
Original file line number Diff line number Diff line change
@@ -0,0 +1,28 @@
import { NgModule } from '@angular/core';
import { CommonModule } from '@angular/common';
import { FilteredCollectionsComponent } from './filtered-collections/filtered-collections.component';
import { RouterModule } from '@angular/router';
import { SharedModule } from '../../shared/shared.module';
import { FormModule } from '../../shared/form/form.module';
import { FilteredItemsComponent } from './filtered-items/filtered-items.component';
import { AdminReportsRoutingModule } from './admin-reports-routing.module';
import { NgbAccordionModule } from '@ng-bootstrap/ng-bootstrap';
import { FiltersComponent } from './filters-section/filters-section.component';

@NgModule({
imports: [
CommonModule,
SharedModule,
RouterModule,
FormModule,
AdminReportsRoutingModule,
NgbAccordionModule
],
declarations: [
FilteredCollectionsComponent,
FilteredItemsComponent,
FiltersComponent
]
})
export class AdminReportsModule {
}
Original file line number Diff line number Diff line change
@@ -0,0 +1,36 @@
export class FilteredCollection {

public label: string;
public handle: string;
public communityLabel: string;
public communityHandle: string;
public nbTotalItems: number;
public values = {};
public allFiltersValue: number;

public clear() {
this.label = '';
this.handle = '';
this.communityLabel = '';
this.communityHandle = '';
this.nbTotalItems = 0;
this.values = {};
this.allFiltersValue = 0;
}

public deserialize(object: any) {
this.clear();
this.label = object.label;
this.handle = object.handle;
this.communityLabel = object.community_label;
this.communityHandle = object.community_handle;
this.nbTotalItems = object.nb_total_items;
let valuesPerFilter = object.values;
for (let filter in valuesPerFilter) {
if (valuesPerFilter.hasOwnProperty(filter)) {
this.values[filter] = valuesPerFilter[filter];
}
}
this.allFiltersValue = object.all_filters_value;
}
}
artlowel marked this conversation as resolved.
Show resolved Hide resolved
Empty file.
Original file line number Diff line number Diff line change
@@ -0,0 +1,64 @@
<div class="container">
<div class="metadata-registry row">
<div class="col-12">

<h2 id="header" class="border-bottom pb-2">{{ "admin.reports.collections.head" | translate }}</h2>
tdonohue marked this conversation as resolved.
Show resolved Hide resolved

<div id="metadatadiv">
<ngb-accordion [closeOthers]="true" activeIds="filters" #acc="ngbAccordion">
<ngb-panel id="filters">
<ng-template ngbPanelTitle>
{{ "admin.reports.commons.filters" | translate }}
</ng-template>
<ng-template ngbPanelContent>
<div class="container">
<div class="row">
<span class="col-3"></span>
<button class="btn btn-light mt-1 col-6" (click)="submit()">{{ "admin.reports.button.show-collections" | translate }}</button>
</div>
<ds-filters [filtersForm]="filtersFormGroup()"></ds-filters>
<div class="row">
<span class="col-3"></span>
<button class="btn btn-light mt-1 col-6" (click)="submit()">{{ "admin.reports.button.show-collections" | translate }}</button>
</div>
</div>
</ng-template>
</ngb-panel>
<ngb-panel id="collections">
<ng-template ngbPanelTitle>
{{ "admin.reports.collections.collections-report" | translate }}
</ng-template>
<ng-template ngbPanelContent>
<table id="table" class="stats">
<thead>
<tr class="header">
<th rowspan="2">{{ "admin.reports.collections.community" | translate }}</th>
<th rowspan="2">{{ "admin.reports.collections.collection" | translate }}</th>
<th>{{ "admin.reports.collections.nb_items" | translate }}</th>
<th>{{ "admin.reports.collections.match_all_selected_filters" | translate }}</th>
<th *ngFor="let filter of results.summary.values | keyvalue">{{ ("admin.reports.commons.filters." + getGroup(filter.key) + "." + filter.key) | translate }}</th>
</tr>
<tr class="header">
<th class="num">{{ results.summary.nbTotalItems }}</th>
<th class="num">{{ results.summary.allFiltersValue }}</th>
<th class="num" *ngFor="let filter of results.summary.values | keyvalue">{{ filter.value }}</th>
</tr>
</thead>
<tbody>
<tr *ngFor="let coll of results.collections">
<td><a href="{{ coll.communityHandle }}" target="_blank">{{ coll.communityLabel }}</a></td>
Fixed Show fixed Hide fixed
tdonohue marked this conversation as resolved.
Show resolved Hide resolved
<td><a href="{{ coll.handle }}" target="_blank">{{ coll.label }}</a></td>
Fixed Show fixed Hide fixed
tdonohue marked this conversation as resolved.
Show resolved Hide resolved
<td class="num">{{ coll.nbTotalItems }}</td>
<td class="num">{{ coll.allFiltersValue }}</td>
<td class="num" *ngFor="let filter of results.summary.values | keyvalue">{{ coll.values[filter.key] || 0 }}</td>
</tr>
</tbody>
</table>
</ng-template>
</ngb-panel>
</ngb-accordion>
</div>

</div>
</div>
</div>
Original file line number Diff line number Diff line change
@@ -0,0 +1,15 @@
.num {
text-align: center;
}

.stats tbody tr:nth-child(even) {
background: #CCC
}
.stats tbody tr:nth-child(odd) {
background: #FFF
}
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Rather than defining odd and even coloring yourself, please use bootstrap classes for striped rows. That way they can be themed

Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Should be OK.

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Done. This simplifies my code indeed.


.stats thead tr {
border-bottom-width: 1px;
border-bottom-style: solid;
}
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Same here, I'd avoid the custom table styling, and use bootstrap's table style instead. It'll make things more consistent

Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Will need some guidance on this one.

Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

If you use the classes table and table-striped on the table the header row will automatically get a pronounced bottom border, as you can see in the examples in the docs

Original file line number Diff line number Diff line change
@@ -0,0 +1,83 @@
import { waitForAsync, ComponentFixture, TestBed} from '@angular/core/testing';
import { NO_ERRORS_SCHEMA } from '@angular/core';
import { TranslateLoader, TranslateModule } from '@ngx-translate/core';
import { TranslateLoaderMock } from 'src/app/shared/mocks/translate-loader.mock';
import { FormBuilder } from '@angular/forms';
import { FilteredCollectionsComponent } from './filtered-collections.component';
import { DspaceRestService } from 'src/app/core/dspace-rest/dspace-rest.service';
import { NgbAccordion, NgbAccordionModule } from '@ng-bootstrap/ng-bootstrap';
import { HttpClientTestingModule } from '@angular/common/http/testing';
import { of as observableOf } from 'rxjs';
import { RawRestResponse } from 'src/app/core/dspace-rest/raw-rest-response.model';

describe('FiltersComponent', () => {
let component: FilteredCollectionsComponent;
let fixture: ComponentFixture<FilteredCollectionsComponent>;
let formBuilder: FormBuilder;

const expected = {
payload: {
collections: [],
summary: {
label: 'Test'
}
},
statusCode: 200,
statusText: 'OK'
} as RawRestResponse;

beforeEach(waitForAsync(() => {
TestBed.configureTestingModule({
declarations: [FilteredCollectionsComponent],
imports: [
NgbAccordionModule,
TranslateModule.forRoot({
loader: {
provide: TranslateLoader,
useClass: TranslateLoaderMock
}
}),
HttpClientTestingModule
],
providers: [
FormBuilder,
DspaceRestService
],
schemas: [NO_ERRORS_SCHEMA]
});
}));

beforeEach(waitForAsync(() => {
formBuilder = TestBed.inject(FormBuilder);

fixture = TestBed.createComponent(FilteredCollectionsComponent);
component = fixture.componentInstance;
fixture.detectChanges();
}));

it('should create the component', () => {
expect(component).toBeTruthy();
});

it('should be displaying the filters panel initially', () => {
let accordion: NgbAccordion = component.accordionComponent;
expect(accordion.isExpanded('filters')).toBeTrue();
});

describe('toggle', () => {
beforeEach(() => {
spyOn(component, 'postFilteredCollections').and.returnValue(observableOf(expected));
spyOn(component.results, 'deserialize');
spyOn(component.accordionComponent, 'expand').and.callThrough();
component.submit();
fixture.detectChanges();
});

it('should be displaying the collections panel after submitting', waitForAsync(() => {
fixture.whenStable().then(() => {
expect(component.accordionComponent.expand).toHaveBeenCalledWith('collections');
expect(component.accordionComponent.isExpanded('collections')).toBeTrue();
});
}));
});
});
Original file line number Diff line number Diff line change
@@ -0,0 +1,58 @@
import { Component, ViewChild } from '@angular/core';
import { FormBuilder, FormGroup } from '@angular/forms';
import { NgbAccordion } from '@ng-bootstrap/ng-bootstrap';
import { Observable } from 'rxjs';
import { RestRequestMethod } from 'src/app/core/data/rest-request-method';
import { DspaceRestService } from 'src/app/core/dspace-rest/dspace-rest.service';
import { RawRestResponse } from 'src/app/core/dspace-rest/raw-rest-response.model';
import { environment } from 'src/environments/environment';
import { FiltersComponent } from '../filters-section/filters-section.component';
import { FilteredCollections } from './filtered-collections.model';

@Component({
selector: 'ds-report-filtered-collections',
templateUrl: './filtered-collections.component.html',
styleUrls: ['./filtered-collections.component.scss']
})
export class FilteredCollectionsComponent {
tdonohue marked this conversation as resolved.
Show resolved Hide resolved

queryForm: FormGroup;
results: FilteredCollections = new FilteredCollections();
@ViewChild('acc') accordionComponent: NgbAccordion;

constructor(
private formBuilder: FormBuilder,
private restService: DspaceRestService) {}

ngOnInit() {
this.queryForm = this.formBuilder.group({
filters: FiltersComponent.formGroup(this.formBuilder)
});
}

filtersFormGroup(): FormGroup {
return this.queryForm.get('filters') as FormGroup;
}

getGroup(filterId: string): string {
return FiltersComponent.getGroup(filterId).id;
}

submit() {
this
.postFilteredCollections()
.subscribe(
response => {
this.results.deserialize(response.payload);
this.accordionComponent.expand('collections');
}
);
}

postFilteredCollections(): Observable<RawRestResponse> {
let form = this.queryForm.value;
let scheme = environment.rest.ssl ? 'https' : 'http';
let urlRestApp = `${scheme}://${environment.rest.host}:${environment.rest.port}${environment.rest.nameSpace}`;
return this.restService.request(RestRequestMethod.POST, `${urlRestApp}/api/contentreport/filteredcollections`, form);
}
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

In order not to duplicate code like this all across the app, and to be able to cache responses, please use a Dataservice for this instead. While a request like this might have been a POST(i.e. a form submit) in dspace 6, with a REST api, POSTs don't make a lot of sense in this context, as you're not creating anything. This should be a GET endpoint. And when it's a GET you can likely get away with simply extending HALDataService<FilteredCollections> in your DataService.

The FilteredCollection model will also need cerialize annotations (basically put @autoSerialize above each property that comes from rest), and then you don't need to (de)serialize it manually anymore either.

Same for FilteredCollections, however because collections and summary are complex types you'll likely need an @autoserializeAs(FilteredCollection[]) for those

You'll also need to add a property for the type (I see that's already in the rest response, just not in the model in angular) and add the @typedObject annotation to the class, a static type property as well as a regular type property, as you can see here

Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

We will need guidance here too. We tried at first to build some DataService, but to no avail, so we decided to use lower-level REST queries directly in the component. If we can move this code to a full-featured DataService, of course it might be better.

}
Original file line number Diff line number Diff line change
@@ -0,0 +1,26 @@
import { FilteredCollection } from './filtered-collection.model';

export class FilteredCollections {

public collections: Array<FilteredCollection> = [];
public summary: FilteredCollection = new FilteredCollection();

public clear() {
this.collections.splice(0, this.collections.length);
this.summary.clear();
}

public deserialize(object: any) {
this.clear();
let summary = object.summary;
this.summary.deserialize(summary);
let collections = object.collections;
for (let i = 0; i < collections.length; i++) {
let collection = collections[i];
let coll = new FilteredCollection();
coll.deserialize(collection);
this.collections.push(coll);
}
}

}
23 changes: 23 additions & 0 deletions src/app/admin/admin-reports/filtered-items/filtered-items-model.ts
Original file line number Diff line number Diff line change
@@ -0,0 +1,23 @@
import { Item } from 'src/app/core/shared/item.model';

export class FilteredItems {

public items: Item[] = [];
public itemCount: number;

public clear() {
this.items.splice(0, this.items.length);
}

public deserialize(object: any, offset: number = 0) {
this.clear();
this.itemCount = object.itemCount;
let items = object.items;
for (let i = 0; i < items.length; i++) {
let item = items[i];
item.index = this.items.length + offset + 1;
this.items.push(item);
}
}

}
Empty file.
Loading