diff --git a/src/app/+item-page/edit-item-page/item-move/item-move.component.html b/src/app/+item-page/edit-item-page/item-move/item-move.component.html
index fe27ed36a5..0c97628d4c 100644
--- a/src/app/+item-page/edit-item-page/item-move/item-move.component.html
+++ b/src/app/+item-page/edit-item-page/item-move/item-move.component.html
@@ -6,7 +6,7 @@
;
+
+const mockItem = Object.assign(new Item(), {
+ id: 'fake-id',
+ handle: 'fake/handle',
+ lastModified: '2018'
+});
+
+const itemPageUrl = `fake-url/${mockItem.id}`;
+const routerStub = Object.assign(new RouterStub(), {
+ url: `${itemPageUrl}/edit`
+});
+
+const mockItemDataService = jasmine.createSpyObj({
+ moveToCollection: Observable.of(new RestResponse(true, '200'))
+});
+
+const mockItemDataServiceFail = jasmine.createSpyObj({
+ moveToCollection: Observable.of(new RestResponse(false, '500'))
+});
+
+const routeStub = {
+ data: Observable.of({
+ item: new RemoteData(false, false, true, null, {
+ id: 'item1'
+ })
+ })
+};
+
+const mockSearchService = {
+ search: () => {
+ return Observable.of(new RemoteData(false, false, true, null,
+ new PaginatedList(null, [
+ {
+ dspaceObject: {
+ name: 'Test collection 1',
+ uuid: 'collection1'
+ }, hitHighlights: {}
+ }, {
+ dspaceObject: {
+ name: 'Test collection 2',
+ uuid: 'collection2'
+ }, hitHighlights: {}
+ }
+ ])));
+ }
+};
+
+const notificationsServiceStub = new NotificationsServiceStub();
+
+describe('ItemMoveComponent', () => {
+ beforeEach(async(() => {
+ TestBed.configureTestingModule({
+ imports: [CommonModule, FormsModule, RouterTestingModule.withRoutes([]), TranslateModule.forRoot(), NgbModule.forRoot()],
+ declarations: [ItemMoveComponent],
+ providers: [
+ {provide: ActivatedRoute, useValue: routeStub},
+ {provide: Router, useValue: routerStub},
+ {provide: ItemDataService, useValue: mockItemDataService},
+ {provide: NotificationsService, useValue: notificationsServiceStub},
+ {provide: SearchService, useValue: mockSearchService},
+ ], schemas: [
+ CUSTOM_ELEMENTS_SCHEMA
+ ]
+ }).compileComponents();
+ }));
+
+ beforeEach(() => {
+ fixture = TestBed.createComponent(ItemMoveComponent);
+ comp = fixture.componentInstance;
+ fixture.detectChanges();
+ });
+ it('should load suggestions', () => {
+ const expected = [
+ {
+ displayValue: 'Test collection 1',
+ value: {
+ name: 'Test collection 1',
+ id: 'collection1',
+ }
+ },
+ {
+ displayValue: 'Test collection 2',
+ value: {
+ name: 'Test collection 2',
+ id: 'collection2',
+ }
+ }
+ ];
+
+ comp.CollectionSearchResults.subscribe((value) => {
+ expect(value).toEqual(expected);
+ }
+ );
+ });
+ it('should get current url ', () => {
+ expect(comp.getCurrentUrl()).toEqual('fake-url/fake-id/edit');
+ });
+ it('should on click select the correct collection name and id', () => {
+ const data = {
+ name: 'Test collection 1',
+ id: 'collection1',
+ };
+ comp.onClick(data);
+
+ expect(comp.selectedCollection).toEqual('Test collection 1');
+ expect(comp.selectedCollectionId).toEqual('collection1');
+ });
+ describe('moveCollection', () => {
+ it('should call itemDataService.moveToCollection', () => {
+ comp.itemId = 'item-id';
+ comp.selectedCollectionId = 'selected-collection-id';
+ comp.moveCollection();
+
+ expect(mockItemDataService.moveToCollection).toHaveBeenCalledWith('item-id', 'selected-collection-id');
+ });
+ it('should call notificationsService success message on success', () => {
+ spyOn(notificationsServiceStub, 'success');
+
+ comp.moveCollection();
+
+ expect(notificationsServiceStub.success).toHaveBeenCalled();
+ });
+ });
+});
+
+describe('ItemMoveComponent fail', () => {
+ beforeEach(async(() => {
+ TestBed.configureTestingModule({
+ imports: [CommonModule, FormsModule, RouterTestingModule.withRoutes([]), TranslateModule.forRoot(), NgbModule.forRoot()],
+ declarations: [ItemMoveComponent],
+ providers: [
+ {provide: ActivatedRoute, useValue: routeStub},
+ {provide: Router, useValue: routerStub},
+ {provide: ItemDataService, useValue: mockItemDataServiceFail},
+ {provide: NotificationsService, useValue: notificationsServiceStub},
+ {provide: SearchService, useValue: mockSearchService},
+ ], schemas: [
+ CUSTOM_ELEMENTS_SCHEMA
+ ]
+ }).compileComponents();
+ }));
+
+ beforeEach(() => {
+ fixture = TestBed.createComponent(ItemMoveComponent);
+ comp = fixture.componentInstance;
+ fixture.detectChanges();
+ });
+
+ it('should call notificationsService error message on fail', () => {
+ spyOn(notificationsServiceStub, 'error');
+
+ comp.moveCollection();
+
+ expect(notificationsServiceStub.error).toHaveBeenCalled();
+ });
+});
diff --git a/src/app/+item-page/edit-item-page/item-move/item-move.component.ts b/src/app/+item-page/edit-item-page/item-move/item-move.component.ts
index e0819257c2..338d4b96a3 100644
--- a/src/app/+item-page/edit-item-page/item-move/item-move.component.ts
+++ b/src/app/+item-page/edit-item-page/item-move/item-move.component.ts
@@ -8,12 +8,9 @@ import {RemoteData} from '../../../core/data/remote-data';
import {DSpaceObject} from '../../../core/shared/dspace-object.model';
import {PaginatedList} from '../../../core/data/paginated-list';
import {SearchResult} from '../../../+search-page/search-result.model';
-import {PaginatedSearchOptions} from '../../../+search-page/paginated-search-options.model';
import {Item} from '../../../core/shared/item.model';
import {ActivatedRoute, Router} from '@angular/router';
import {NotificationsService} from '../../../shared/notifications/notifications.service';
-import {CollectionDataService} from '../../../core/data/collection-data.service';
-import {SearchConfigurationService} from '../../../+search-page/search-service/search-configuration.service';
import {TranslateService} from '@ngx-translate/core';
import {getSucceededRemoteData} from '../../../core/shared/operators';
import {ItemDataService} from '../../../core/data/item-data.service';
@@ -24,15 +21,14 @@ import {getItemEditPath} from '../../item-page-routing.module';
selector: 'ds-item-move',
templateUrl: './item-move.component.html'
})
+/**
+ * Component that handles the moving of an item to a different collection
+ */
export class ItemMoveComponent implements OnInit {
inheritPolicies = false;
itemRD$: Observable>;
- /**
- * Search options
- */
- searchOptions$: Observable;
- filterSearchResults: Observable = Observable.of([]);
+ CollectionSearchResults: Observable = Observable.of([]);
selectedCollection: string;
selectedCollectionId: string;
@@ -41,9 +37,7 @@ export class ItemMoveComponent implements OnInit {
constructor(private route: ActivatedRoute,
private router: Router,
private notificationsService: NotificationsService,
- private collectionDataService: CollectionDataService,
private itemDataService: ItemDataService,
- private searchConfigService: SearchConfigurationService,
private searchService: SearchService,
private translateService: TranslateService) {
}
@@ -54,10 +48,13 @@ export class ItemMoveComponent implements OnInit {
this.itemId = rd.payload.id;
}
);
- this.searchOptions$ = this.searchConfigService.paginatedSearchOptions;
this.loadSuggestions('');
}
+ /**
+ * Find suggestions based on entered query
+ * @param query - Search query
+ */
findSuggestions(query): void {
this.loadSuggestions(query);
}
@@ -67,7 +64,7 @@ export class ItemMoveComponent implements OnInit {
* TODO: When the API support it, only fetch collections where user has ADD rights to.
*/
loadSuggestions(query): void {
- this.filterSearchResults = this.searchService.search(new SearchOptions({
+ this.CollectionSearchResults = this.searchService.search(new SearchOptions({
dsoType: DSpaceObjectType.COLLECTION,
query: query
})).first().pipe(
@@ -83,6 +80,10 @@ export class ItemMoveComponent implements OnInit {
}
+ /**
+ * Set the collection name and id based on the selected value
+ * @param data - obtained from the ds-input-suggestions component
+ */
onClick(data: any): void {
this.selectedCollection = data.name;
this.selectedCollectionId = data.id;
@@ -95,6 +96,9 @@ export class ItemMoveComponent implements OnInit {
return this.router.url;
}
+ /**
+ * Moves the item to a new collection based on the selected collection
+ */
moveCollection() {
this.itemDataService.moveToCollection(this.itemId, this.selectedCollectionId).first().subscribe(
(response: RestResponse) => {
diff --git a/src/app/+item-page/edit-item-page/item-operation/item-operation.component.spec.ts b/src/app/+item-page/edit-item-page/item-operation/item-operation.component.spec.ts
new file mode 100644
index 0000000000..092f3af0ac
--- /dev/null
+++ b/src/app/+item-page/edit-item-page/item-operation/item-operation.component.spec.ts
@@ -0,0 +1,44 @@
+import {ItemOperation} from './itemOperation.model';
+import {async, TestBed} from '@angular/core/testing';
+import {ItemOperationComponent} from './item-operation.component';
+import {TranslateModule} from '@ngx-translate/core';
+import {By} from '@angular/platform-browser';
+
+describe('ItemOperationComponent', () => {
+ const itemOperation: ItemOperation = new ItemOperation('key1', 'url1');
+
+ let fixture;
+ let comp;
+
+ beforeEach(async(() => {
+ TestBed.configureTestingModule({
+ imports: [TranslateModule.forRoot()],
+ declarations: [ItemOperationComponent]
+ }).compileComponents();
+ }));
+
+ beforeEach(() => {
+
+ fixture = TestBed.createComponent(ItemOperationComponent);
+ comp = fixture.componentInstance;
+ comp.operation = itemOperation;
+ fixture.detectChanges();
+ });
+
+ it('should render operation row', () => {
+ const span = fixture.debugElement.query(By.css('span')).nativeElement;
+ expect(span.textContent).toContain('item.edit.tabs.status.buttons.key1.label');
+ const link = fixture.debugElement.query(By.css('a')).nativeElement;
+ expect(link.href).toContain('url1');
+ expect(link.textContent).toContain('item.edit.tabs.status.buttons.key1.button');
+ });
+ it('should render disabled operation row', () => {
+ itemOperation.setDisabled(true);
+ fixture.detectChanges();
+
+ const span = fixture.debugElement.query(By.css('span')).nativeElement;
+ expect(span.textContent).toContain('item.edit.tabs.status.buttons.key1.label');
+ const span2 = fixture.debugElement.query(By.css('span.btn-danger')).nativeElement;
+ expect(span2.textContent).toContain('item.edit.tabs.status.buttons.key1.button');
+ });
+});
diff --git a/src/app/+item-page/edit-item-page/item-operation/item-operation.component.ts b/src/app/+item-page/edit-item-page/item-operation/item-operation.component.ts
index 951d66cbd8..76d056df95 100644
--- a/src/app/+item-page/edit-item-page/item-operation/item-operation.component.ts
+++ b/src/app/+item-page/edit-item-page/item-operation/item-operation.component.ts
@@ -5,7 +5,9 @@ import {ItemOperation} from './itemOperation.model';
selector: 'ds-item-operation',
templateUrl: './item-operation.component.html'
})
-
+/**
+ * Operation that can be performed on an item
+ */
export class ItemOperationComponent {
@Input() operation: ItemOperation;
diff --git a/src/app/+item-page/edit-item-page/item-operation/itemOperation.model.ts b/src/app/+item-page/edit-item-page/item-operation/itemOperation.model.ts
index 6a54744fcb..0104dfbdb3 100644
--- a/src/app/+item-page/edit-item-page/item-operation/itemOperation.model.ts
+++ b/src/app/+item-page/edit-item-page/item-operation/itemOperation.model.ts
@@ -7,6 +7,15 @@ export class ItemOperation {
constructor(operationKey: string, operationUrl: string) {
this.operationKey = operationKey;
this.operationUrl = operationUrl;
+ this.setDisabled(false);
+ }
+
+ /**
+ * Set whether this operation should be disabled
+ * @param disabled
+ */
+ setDisabled(disabled: boolean): void {
+ this.disabled = disabled;
}
}
diff --git a/src/app/+item-page/edit-item-page/item-status/item-status.component.spec.ts b/src/app/+item-page/edit-item-page/item-status/item-status.component.spec.ts
index 2df4b977cb..319d4c47ae 100644
--- a/src/app/+item-page/edit-item-page/item-status/item-status.component.spec.ts
+++ b/src/app/+item-page/edit-item-page/item-status/item-status.component.spec.ts
@@ -10,6 +10,7 @@ import { Router } from '@angular/router';
import { RouterStub } from '../../../shared/testing/router-stub';
import { Item } from '../../../core/shared/item.model';
import { By } from '@angular/platform-browser';
+import {CUSTOM_ELEMENTS_SCHEMA} from '@angular/core';
describe('ItemStatusComponent', () => {
let comp: ItemStatusComponent;
@@ -33,7 +34,7 @@ describe('ItemStatusComponent', () => {
providers: [
{ provide: Router, useValue: routerStub },
{ provide: HostWindowService, useValue: new HostWindowServiceStub(0) }
- ]
+ ], schemas: [CUSTOM_ELEMENTS_SCHEMA]
}).compileComponents();
}));
diff --git a/src/app/+search-page/search-filters/search-filter/search-hierarchy-filter/search-hierarchy-filter.component.html b/src/app/+search-page/search-filters/search-filter/search-hierarchy-filter/search-hierarchy-filter.component.html
index 812f543716..962d09e6c4 100644
--- a/src/app/+search-page/search-filters/search-filter/search-hierarchy-filter/search-hierarchy-filter.component.html
+++ b/src/app/+search-page/search-filters/search-filter/search-hierarchy-filter/search-hierarchy-filter.component.html
@@ -30,7 +30,7 @@
| translate}}
-
-