diff --git a/project-manager/karma.conf.js b/project-manager/karma.conf.js index 9e88999..9f0e343 100644 --- a/project-manager/karma.conf.js +++ b/project-manager/karma.conf.js @@ -1,5 +1,5 @@ // Karma configuration -// Generated on Sat Mar 25 2017 06:31:51 GMT+0100 (CET) +// Generated on Thu Mar 09 2017 06:31:51 GMT+0100 (CET) module.exports = function(config) { config.set({ @@ -12,6 +12,9 @@ module.exports = function(config) { // available frameworks: https://npmjs.org/browse/keyword/karma-adapter frameworks: ['jasmine', '@angular/cli'], + + // list of plugins to load. A plugin can be a string (in which case it will + // be required by Karma) or an inlined plugin - Object plugins: [ require('karma-jasmine'), require('karma-chrome-launcher'), diff --git a/project-manager/src/app/blog/blog-entry/blog-entry.component.html b/project-manager/src/app/blog/blog-entry/blog-entry.component.html index 22a4e2f..e7c99b9 100644 --- a/project-manager/src/app/blog/blog-entry/blog-entry.component.html +++ b/project-manager/src/app/blog/blog-entry/blog-entry.component.html @@ -4,9 +4,14 @@
{{entry.title}} -

{{entry.text}}

+

{{entry.text}}

+
+
+ created at: + {{entry.createdAt | date: 'yyyy-MM-dd'}} +
diff --git a/project-manager/src/app/blog/blog-entry/blog-entry.component.spec.ts b/project-manager/src/app/blog/blog-entry/blog-entry.component.spec.ts new file mode 100644 index 0000000..86a7a58 --- /dev/null +++ b/project-manager/src/app/blog/blog-entry/blog-entry.component.spec.ts @@ -0,0 +1,60 @@ +import {TestBed} from "@angular/core/testing"; +import {BlogEntryComponent} from "./blog-entry.component"; +import {BlogEntry} from "./blog-entry"; + +describe('Blog Entry Isolated Test', () => { + + beforeEach(() => { + TestBed.configureTestingModule({ + declarations: [BlogEntryComponent], + }) + }); + + it('should render DOM correct according to Input', () => { + // Umgebung initialisieren + const fixture = TestBed.createComponent(BlogEntryComponent); + const blogEntryComponent : BlogEntryComponent = fixture.componentInstance; + const element = fixture.nativeElement; + const blogEntry : BlogEntry = new BlogEntry; + + // Testdaten + const testId = 101; + const testTitle = "test title"; + const testText = "test text"; + const testImage = "https://www.google.de/images/branding/googlelogo/2x/googlelogo_color_272x92dp.png"; + + // Input-Daten + blogEntry.id = testId; + blogEntry.title = testTitle; + blogEntry.text = testText; + blogEntry.image = testImage; + blogEntry.createdAt = new Date(); + ( blogEntryComponent).entry = blogEntry; + + // DOM update + fixture.detectChanges(); + + // DOM auslesen und Daten abgleichen + const imageSrc = element.querySelector('div /deep/ .blog-image > img').getAttribute("src"); + expect(imageSrc).toBe(testImage); + + // textContent statt innerHtml + // siehe http://stackoverflow.com/questions/40227533/angular-2-and-jasmine-unit-testing-cannot-get-the-innerhtml + const blogTitle = element.querySelector('div /deep/ .blog-summary > span'); + expect(blogTitle.textContent).toBe(testTitle); + + const blogText = element.querySelector('div /deep/ .blog-summary > p').textContent; + expect(blogText).toBe(testText); + + const blogTs = element.querySelector('div /deep/ .blog-timestamp > .timestamp').textContent; + expect(new Date(blogTs).getDate()).toBe(new Date().getDate()); + + const blogDelete = element.querySelector('div /deep/ .blog-delete > button').textContent; + expect(blogDelete).toBe("Entfernen") + + // keine Überprüfung der id, die lediglich im EventListener des Delete Button hinterlegt ist, möglich! + // nur durch click-Event auslösen, was jedoch BlogComponent vorraussetzt + + }); + +}); diff --git a/project-manager/src/app/blog/blog-entry/blog-entry.component.ts b/project-manager/src/app/blog/blog-entry/blog-entry.component.ts index e348332..933e4aa 100644 --- a/project-manager/src/app/blog/blog-entry/blog-entry.component.ts +++ b/project-manager/src/app/blog/blog-entry/blog-entry.component.ts @@ -1,4 +1,4 @@ -import {Component, Input } from '@angular/core'; +import {Component, Input} from '@angular/core'; import {BlogEntry} from './blog-entry'; import {BlogComponent} from "../blog.component"; diff --git a/project-manager/src/app/blog/blog-entry/blog-entry.ts b/project-manager/src/app/blog/blog-entry/blog-entry.ts index 087336f..e37f920 100644 --- a/project-manager/src/app/blog/blog-entry/blog-entry.ts +++ b/project-manager/src/app/blog/blog-entry/blog-entry.ts @@ -1,6 +1,7 @@ export class BlogEntry { - id :number; + id : number; title: string; text: string; image: string; + createdAt: Date; } diff --git a/project-manager/src/app/blog/blog.component.spec.ts b/project-manager/src/app/blog/blog.component.spec.ts index c562ebf..ab216bb 100644 --- a/project-manager/src/app/blog/blog.component.spec.ts +++ b/project-manager/src/app/blog/blog.component.spec.ts @@ -1,9 +1,48 @@ import {BlogComponent} from './blog.component' +import {Title} from "@angular/platform-browser"; -describe('', () => { +describe('Blog Component Isolated Test', () => { - it('', () => { + let blogComponent: BlogComponent; + beforeEach(() => { + blogComponent = new BlogComponent(null, null, new Title('Project Blog')); }); + it('should have initial entries', () => { + expect(blogComponent.entries.length).toBe(2); + blogComponent.entries.forEach((entry) => { + expect(entry.id).toBeLessThanOrEqual(blogComponent.id); + expect(entry.createdAt.getDate()).toBe(new Date().getDate()); + }) + }); + + it('should create new list entry and increment id', () => { + let preCreationId = blogComponent.id; + let entryTitle = "some fancy title"; + let entryImage = "https://avatars1.githubusercontent.com/u/3284117"; + let entryText = "some important text"; + blogComponent.createBlogEntry(entryTitle, entryImage, entryText); + + let newEntry = blogComponent.entries[blogComponent.entries.length - 1]; + expect(newEntry.id - 1).toBe(preCreationId) + expect(newEntry.image).toBe(entryImage); + expect(newEntry.text).toBe(entryText); + expect(newEntry.createdAt.getDate()).toBe(new Date().getDate()); + }); + + it('should delete entry by given id - and not change global max-id', () => { + let preDeletionId = blogComponent.id; + let latestId = blogComponent.entries[blogComponent.entries.length - 1].id; + blogComponent.deleteBlogEntry(latestId); + + expect(blogComponent.id).toBe(preDeletionId); + expect(() => { + if (blogComponent.entries.length > 0) { + return blogComponent.entries[blogComponent.entries.length - 1]; + } else { + return 0; + } + }).not.toBe(latestId); + }); }); diff --git a/project-manager/src/app/blog/blog.component.ts b/project-manager/src/app/blog/blog.component.ts index 920742a..acabd6f 100644 --- a/project-manager/src/app/blog/blog.component.ts +++ b/project-manager/src/app/blog/blog.component.ts @@ -18,9 +18,12 @@ export class BlogComponent { constructor(r: ActivatedRoute, private router: Router, private titleService: Title) { this.entries = initialEntries; - if (this.entries.length != 0) { - this.id = this.entries[this.entries.length-1].id; - } + this.entries.forEach((entry)=> { + entry.createdAt = new Date() + if (this.id < entry.id) { + this.id = entry.id; + } + }); } createBlogEntry(title:string, image:string, text:string) { @@ -31,6 +34,8 @@ export class BlogComponent { entry.title = title; entry.image = image; entry.text = text; + entry.createdAt = new Date(); + console.log(this.entries); this.entries.push(entry); } diff --git a/project-manager/src/app/blog/initialEntries.ts b/project-manager/src/app/blog/initialEntries.ts index 07dcb67..41700b9 100644 --- a/project-manager/src/app/blog/initialEntries.ts +++ b/project-manager/src/app/blog/initialEntries.ts @@ -3,12 +3,14 @@ export var initialEntries = [ id: 1, title: "Angular 2 erschienen", image: "", - text: "Endlich ist die finale Version von Angular 2 erschienen. Nach langer Beta-Phase ..." + text: "Endlich ist die finale Version von Angular 2 erschienen. Nach langer Beta-Phase ...", + createdAt: null }, { id: 2, title: "Neue Vorschläge für die kommende Module-Loader-Syntax", image: "https://cdn-images-1.medium.com/max/1000/1*RQFLsbQumE-iNrXzs_Oz8g.jpeg", - text: "Nachdem im ES2015-Standard bereits die Module-API verabschiedet wurde, hat das zuständige Konsortium nun neue Vorschläge für die Module-Loader-Syntax bekannt gegeben..." + text: "Nachdem im ES2015-Standard bereits die Module-API verabschiedet wurde, hat das zuständige Konsortium nun neue Vorschläge für die Module-Loader-Syntax bekannt gegeben...", + createdAt : null } -] \ No newline at end of file +];