From e0ee8b5cad9d9627d7b586becc39b19a7fa97aec Mon Sep 17 00:00:00 2001
From: Ramon Wenger
Date: Tue, 31 Aug 2021 16:39:55 +0200
Subject: [PATCH 01/28] Add failing tests for new features
---
.../integration/frontend/new-project.spec.js | 57 -------
.../frontend/portfolio/new-project.spec.js | 56 +++++++
.../frontend/portfolio/project-entry.spec.js | 157 ++++++++++++++++++
.../frontend/portfolio/projects-page.spec.js | 50 ++++++
.../frontend/project-entry.spec.js | 151 -----------------
5 files changed, 263 insertions(+), 208 deletions(-)
delete mode 100644 client/cypress/integration/frontend/new-project.spec.js
create mode 100644 client/cypress/integration/frontend/portfolio/new-project.spec.js
create mode 100644 client/cypress/integration/frontend/portfolio/project-entry.spec.js
create mode 100644 client/cypress/integration/frontend/portfolio/projects-page.spec.js
delete mode 100644 client/cypress/integration/frontend/project-entry.spec.js
diff --git a/client/cypress/integration/frontend/new-project.spec.js b/client/cypress/integration/frontend/new-project.spec.js
deleted file mode 100644
index 905ee28a..00000000
--- a/client/cypress/integration/frontend/new-project.spec.js
+++ /dev/null
@@ -1,57 +0,0 @@
-const operations = {
- ProjectsQuery: {
- projects: {
- edges: [
- {
- node: {
- id: 'UHJvamVjdE5vZGU6NjY=',
- title: 'Some random title',
- appearance: 'blue',
- description: 'This description rocks',
- slug: 'some-random-title',
- objectives: 'Git gud',
- final: false,
- student: {
- firstName: 'Rachel',
- lastName: 'Green',
- id: 'VXNlck5vZGU6NQ==',
- avatarUrl: '',
- },
- entriesCount: 0,
- },
- },
- ],
- },
- },
- MeQuery: {
- me: {
- }
- },
- AddProject: variables => ({
- addProject: {
- project: Object.assign({}, variables.input.project),
- errors: null,
- __typename: 'AddProjectPayload',
- },
- }),
-};
-
-describe('New project', () => {
- before(() => {
- cy.setup();
- });
-
- it('creates a new project and displays it', () => {
- cy.mockGraphqlOps({
- operations
- });
- cy.visit('/portfolio');
-
- cy.get('[data-cy=add-project-button]').click();
- cy.get('[data-cy=page-form-input-titel]').type('Some random title');
- cy.get('[data-cy=page-form-input-beschreibung]').type('This description rocks');
- cy.get('[data-cy=page-form-input-ziele]').type('Git gud');
- cy.get('[data-cy=save-project-button]').click();
- cy.get('.project-widget:first').contains('random');
- });
-});
diff --git a/client/cypress/integration/frontend/portfolio/new-project.spec.js b/client/cypress/integration/frontend/portfolio/new-project.spec.js
new file mode 100644
index 00000000..86c43062
--- /dev/null
+++ b/client/cypress/integration/frontend/portfolio/new-project.spec.js
@@ -0,0 +1,56 @@
+describe('New project', () => {
+ const operations = {
+ ProjectsQuery: {
+ projects: {
+ edges: [
+ {
+ node: {
+ id: 'UHJvamVjdE5vZGU6NjY=',
+ title: 'Some random title',
+ appearance: 'blue',
+ description: 'This description rocks',
+ slug: 'some-random-title',
+ objectives: 'Git gud',
+ final: false,
+ student: {
+ firstName: 'Rachel',
+ lastName: 'Green',
+ id: 'VXNlck5vZGU6NQ==',
+ avatarUrl: '',
+ },
+ entriesCount: 0,
+ },
+ },
+ ],
+ },
+ },
+ MeQuery: {
+ me: {},
+ },
+ AddProject: variables => ({
+ addProject: {
+ project: Object.assign({}, variables.input.project),
+ errors: null,
+ __typename: 'AddProjectPayload',
+ },
+ }),
+ };
+
+ before(() => {
+ cy.setup();
+ });
+
+ it('creates a new project and displays it', () => {
+ cy.mockGraphqlOps({
+ operations,
+ });
+ cy.visit('/portfolio');
+
+ cy.get('[data-cy=add-project-button]').click();
+ cy.get('[data-cy=page-form-input-titel]').type('Some random title');
+ cy.get('[data-cy=page-form-input-beschreibung]').should('exist').should('be.empty');
+ cy.get('[data-cy=page-form-input-ziele]').should('not.exist');
+ cy.get('[data-cy=save-project-button]').click();
+ cy.get('.project-widget:first').contains('random');
+ });
+});
diff --git a/client/cypress/integration/frontend/portfolio/project-entry.spec.js b/client/cypress/integration/frontend/portfolio/project-entry.spec.js
new file mode 100644
index 00000000..a7ca7a85
--- /dev/null
+++ b/client/cypress/integration/frontend/portfolio/project-entry.spec.js
@@ -0,0 +1,157 @@
+describe('Project Entry', () => {
+ const operations = {
+ MeQuery: {
+ me: {
+ id: 'VXNlck5vZGU6NQ==',
+ permissions: [],
+ },
+ },
+ ProjectsQuery: {
+ projects: {
+ edges: [{
+ node: {
+ id: 'UHJvamVjdE5vZGU6MzM=',
+ title: 'Groot',
+ appearance: 'red',
+ 'description': 'I am Groot',
+ 'slug': 'groot',
+ 'objectives': 'Be Groot\nBe awesome',
+ 'final': false,
+ 'student': {
+ 'firstName': 'Rachel',
+ 'lastName': 'Green',
+ 'id': 'VXNlck5vZGU6NQ==',
+ 'avatarUrl': '',
+ '__typename': 'UserNode',
+ },
+ 'entriesCount': 2,
+ '__typename': 'ProjectNode',
+ },
+ '__typename': 'ProjectNodeEdge',
+ }],
+ '__typename': 'ProjectNodeConnection',
+ },
+ },
+ ProjectQuery: {
+ 'project': {
+ 'id': 'UHJvamVjdE5vZGU6MzY=',
+ 'title': 'Groot',
+ 'appearance': 'yellow',
+ 'description': 'I am Groot',
+ 'slug': 'groot',
+ 'objectives': 'Be Groot\nBe awesome',
+ 'final': false,
+ 'student': {
+ 'firstName': 'Rachel',
+ 'lastName': 'Green',
+ 'id': 'VXNlck5vZGU6NQ==',
+ 'avatarUrl': '',
+ '__typename': 'UserNode',
+ },
+ 'entriesCount': 1,
+ '__typename': 'ProjectNode',
+ 'entries': {
+ 'edges': [{
+ 'node': {
+ 'id': 'UHJvamVjdEVudHJ5Tm9kZTo2NQ==',
+ 'activity': 'Kill Thanos',
+ 'reflection': 'He sucks',
+ 'nextSteps': 'Go for the head',
+ 'documentUrl': '',
+ '__typename': 'ProjectEntryNode',
+ 'created': '2020-01-20T15:20:31.262510+00:00',
+ },
+ '__typename': 'ProjectEntryNodeEdge',
+ }],
+ '__typename': 'ProjectEntryNodeConnection',
+ },
+ },
+ },
+ AddProjectEntry: variables => ({
+ addProjectEntry: {
+ projectEntry: Object.assign({}, variables.input.projectEntry, {
+ created: '2020-01-20T15:26:58.722773+00:00',
+ }),
+ errors: null,
+ __typename: 'AddProjectEntryPayload',
+ },
+ }),
+ UpdateProjectEntry: variables => ({
+ updateProjectEntry: {
+ projectEntry: variables.input.projectEntry,
+ errors: null,
+ __typename: 'UpdateProjectEntryPayload',
+ },
+ }),
+ DeleteProjectEntry: {
+ deleteProjectEntry: {
+ success: true,
+ __typename: 'DeleteProjectEntryPayload',
+ },
+ },
+ };
+
+ beforeEach(() => {
+ cy.setup();
+
+ cy.task('getSchema').then(schema => {
+ cy.mockGraphqlOps({
+ operations,
+ });
+ });
+ });
+
+ it('should create a new project entry', () => {
+ cy.visit('/portfolio');
+ cy.get('[data-cy=project-link]:first-of-type').click();
+ cy.get('[data-cy=add-project-entry]:first-of-type').click();
+ cy.getByDataCy('activity-input').should('not.exist');
+ // cy.get('[data-cy=activity-input]').within(() => {
+ // cy.get('[data-cy=text-form-input]').type('Join the Guardians');
+ // });
+ cy.getByDataCy('reflection-input').should('not.exist');
+ // cy.get('[data-cy=reflection-input]').within(() => {
+ // cy.get('[data-cy=text-form-input]').type('They are cool!');
+ // });
+ cy.getByDataCy('next-steps-input').should('not.exist');
+ // cy.get('[data-cy=next-steps-input]').within(() => {
+ // cy.get('[data-cy=text-form-input]').type('Stay with Rocket\nMeet Quill');
+ // });
+ cy.getByDataCy('modal-title').should('contain', 'Beitrag erfassen');
+ cy.getByDataCy('project-entry-input').should('exist');
+ cy.getByDataCy('use-template-button').should('exist').click();
+ cy.getByDataCy('upload-document-button').should('exist');
+ cy.getByDataCy('modal-save-button').click();
+
+ cy.get('.project-entry:last-of-type').within(() => {
+ cy.get('.project-entry__paragraph:first-of-type').contains('Join the Guardians');
+ });
+ });
+
+ it('should edit first entry', () => {
+ cy.visit('/portfolio/groot');
+ cy.get('.project-entry__paragraph:first-of-type').contains('Kill Thanos');
+ cy.get('.project-entry:first-of-type').within(() => {
+ cy.get('[data-cy=project-entry-more]').click();
+ cy.get('[data-cy=edit-project-entry]').click();
+ });
+ cy.get('[data-cy=activity-input]').within(() => {
+ cy.get('[data-cy=text-form-input]').clear().type('Defeat Thanos');
+ });
+ cy.get('[data-cy=modal-save-button]').click();
+ cy.get('.project-entry__paragraph:first-of-type').contains('Defeat Thanos');
+ });
+
+ it('should delete the last entry', () => {
+ cy.visit('/portfolio/groot');
+
+ cy.get('.project-entry').should('have.length', 1);
+
+ cy.get('.project-entry:last-of-type').within(() => {
+ cy.get('[data-cy=project-entry-more]').click();
+ cy.get('[data-cy=delete-project-entry]').click();
+ });
+
+ cy.get('.project-entry').should('have.length', 0);
+ });
+});
diff --git a/client/cypress/integration/frontend/portfolio/projects-page.spec.js b/client/cypress/integration/frontend/portfolio/projects-page.spec.js
new file mode 100644
index 00000000..623f3d55
--- /dev/null
+++ b/client/cypress/integration/frontend/portfolio/projects-page.spec.js
@@ -0,0 +1,50 @@
+import {getMinimalMe} from '../../../support/helpers';
+
+describe('Projects page', () => {
+ const MeQuery = getMinimalMe({});
+ beforeEach(() => {
+ cy.setup();
+ });
+
+ it('displays portfolio onboarding', () => {
+ const operations = {
+ MeQuery,
+ ProjectsQuery: {
+ projects: {
+ edges: [],
+ },
+ },
+ };
+ cy.mockGraphqlOps({
+ operations,
+ });
+ cy.visit('/portfolio');
+ cy.getByDataCy('page-title').should('contain', 'Portfolio');
+ cy.getByDataCy('portfolio-onboarding-illustration').should('exist');
+ cy.getByDataCy('portfolio-onboarding-subtitle').should('contain', 'Woran denken Sie gerade');
+ cy.getByDataCy('portfolio-onboarding-text').should('contain', 'Hier können Sie Projekte erstellen.');
+ cy.getByDataCy('page-footer').should('not.exist');
+ cy.getByDataCy('create-project-button').should('exist');
+ });
+
+ it('displays the project list', () => {
+ const operations = {
+ MeQuery,
+ ProjectsQuery: {
+ projects: {
+ edges: [],
+ },
+ },
+ };
+
+ cy.mockGraphqlOps({
+ operations,
+ });
+ cy.visit('/portfolio');
+ cy.getByDataCy('page-title').should('contain', 'Portfolio');
+ cy.getByDataCy('create-project-button').should('exist');
+ cy.getByDataCy('project-list').should('exist');
+ cy.getByDataCy('project').should('have.length', 1);
+ cy.getByDataCy('project-owner').should('contain', 'Bilbo Baggins');
+ });
+});
diff --git a/client/cypress/integration/frontend/project-entry.spec.js b/client/cypress/integration/frontend/project-entry.spec.js
deleted file mode 100644
index 6fcfd2c7..00000000
--- a/client/cypress/integration/frontend/project-entry.spec.js
+++ /dev/null
@@ -1,151 +0,0 @@
-
-const operations = {
- MeQuery: {
- me: {
- id: 'VXNlck5vZGU6NQ==',
- permissions: [],
- },
- },
- ProjectsQuery: {
- projects: {
- edges: [{
- node: {
- id: 'UHJvamVjdE5vZGU6MzM=',
- title: 'Groot',
- appearance: 'red',
- 'description': 'I am Groot',
- 'slug': 'groot',
- 'objectives': 'Be Groot\nBe awesome',
- 'final': false,
- 'student': {
- 'firstName': 'Rachel',
- 'lastName': 'Green',
- 'id': 'VXNlck5vZGU6NQ==',
- 'avatarUrl': '',
- '__typename': 'UserNode',
- },
- 'entriesCount': 2,
- '__typename': 'ProjectNode',
- },
- '__typename': 'ProjectNodeEdge',
- }],
- '__typename': 'ProjectNodeConnection',
- },
- },
- ProjectQuery: {
- 'project': {
- 'id': 'UHJvamVjdE5vZGU6MzY=',
- 'title': 'Groot',
- 'appearance': 'yellow',
- 'description': 'I am Groot',
- 'slug': 'groot',
- 'objectives': 'Be Groot\nBe awesome',
- 'final': false,
- 'student': {
- 'firstName': 'Rachel',
- 'lastName': 'Green',
- 'id': 'VXNlck5vZGU6NQ==',
- 'avatarUrl': '',
- '__typename': 'UserNode',
- },
- 'entriesCount': 1,
- '__typename': 'ProjectNode',
- 'entries': {
- 'edges': [{
- 'node': {
- 'id': 'UHJvamVjdEVudHJ5Tm9kZTo2NQ==',
- 'activity': 'Kill Thanos',
- 'reflection': 'He sucks',
- 'nextSteps': 'Go for the head',
- 'documentUrl': '',
- '__typename': 'ProjectEntryNode',
- 'created': '2020-01-20T15:20:31.262510+00:00',
- },
- '__typename': 'ProjectEntryNodeEdge',
- }],
- '__typename': 'ProjectEntryNodeConnection',
- },
- },
- },
- AddProjectEntry: variables => ({
- addProjectEntry: {
- projectEntry: Object.assign({}, variables.input.projectEntry, {
- created: '2020-01-20T15:26:58.722773+00:00',
- }),
- errors: null,
- __typename: 'AddProjectEntryPayload',
- },
- }),
- UpdateProjectEntry: variables => ({
- updateProjectEntry: {
- projectEntry: variables.input.projectEntry,
- errors: null,
- __typename: 'UpdateProjectEntryPayload',
- },
- }),
- DeleteProjectEntry: {
- deleteProjectEntry: {
- success: true,
- __typename: 'DeleteProjectEntryPayload',
- },
- },
-};
-
-describe('Project Entry', () => {
- beforeEach(() => {
- cy.setup();
-
- cy.task('getSchema').then(schema => {
- cy.mockGraphqlOps({
- operations,
- });
- });
- });
-
- it('should create a new project entry', () => {
- cy.visit('/portfolio');
- cy.get('[data-cy=project-link]:first-of-type').click();
- cy.get('[data-cy=add-project-entry]:first-of-type').click();
- cy.get('[data-cy=activity-input]').within(() => {
- cy.get('[data-cy=text-form-input]').type('Join the Guardians');
- });
- cy.get('[data-cy=reflection-input]').within(() => {
- cy.get('[data-cy=text-form-input]').type('They are cool!');
- });
- cy.get('[data-cy=next-steps-input]').within(() => {
- cy.get('[data-cy=text-form-input]').type('Stay with Rocket\nMeet Quill');
- });
- cy.get('[data-cy=modal-save-button]').click();
-
- cy.get('.project-entry:last-of-type').within(() => {
- cy.get('.project-entry__paragraph:first-of-type').contains('Join the Guardians');
- });
- });
-
- it('should edit first entry', () => {
- cy.visit('/portfolio/groot');
- cy.get('.project-entry__paragraph:first-of-type').contains('Kill Thanos');
- cy.get('.project-entry:first-of-type').within(() => {
- cy.get('[data-cy=project-entry-more]').click();
- cy.get('[data-cy=edit-project-entry]').click();
- });
- cy.get('[data-cy=activity-input]').within(() => {
- cy.get('[data-cy=text-form-input]').clear().type('Defeat Thanos');
- });
- cy.get('[data-cy=modal-save-button]').click();
- cy.get('.project-entry__paragraph:first-of-type').contains('Defeat Thanos');
- });
-
- it('should delete the last entry', () => {
- cy.visit('/portfolio/groot');
-
- cy.get('.project-entry').should('have.length', 1);
-
- cy.get('.project-entry:last-of-type').within(() => {
- cy.get('[data-cy=project-entry-more]').click();
- cy.get('[data-cy=delete-project-entry]').click();
- });
-
- cy.get('.project-entry').should('have.length', 0);
- });
-});
From 3f5d70587720b0cc2d3a5f40e43d0338b5744935 Mon Sep 17 00:00:00 2001
From: Ramon Wenger
Date: Tue, 31 Aug 2021 21:22:45 +0200
Subject: [PATCH 02/28] Add portfolio onboarding page
---
.../frontend/portfolio/projects-page.spec.js | 2 +-
.../illustrations/PortfolioIllustration.vue | 347 ++++++------
.../illustrations/RoomsIllustration.vue | 526 +++++++-----------
.../portfolio/PortfolioOnboarding.vue | 51 ++
.../src/components/rooms/RoomsOnboarding.vue | 19 +-
.../src/graphql/gql/queries/allProjects.gql | 6 +-
client/src/pages/portfolio/portfolio.vue | 35 +-
client/src/styles/_mixins.scss | 26 +
8 files changed, 471 insertions(+), 541 deletions(-)
create mode 100644 client/src/components/portfolio/PortfolioOnboarding.vue
diff --git a/client/cypress/integration/frontend/portfolio/projects-page.spec.js b/client/cypress/integration/frontend/portfolio/projects-page.spec.js
index 623f3d55..4c580227 100644
--- a/client/cypress/integration/frontend/portfolio/projects-page.spec.js
+++ b/client/cypress/integration/frontend/portfolio/projects-page.spec.js
@@ -22,7 +22,7 @@ describe('Projects page', () => {
cy.getByDataCy('page-title').should('contain', 'Portfolio');
cy.getByDataCy('portfolio-onboarding-illustration').should('exist');
cy.getByDataCy('portfolio-onboarding-subtitle').should('contain', 'Woran denken Sie gerade');
- cy.getByDataCy('portfolio-onboarding-text').should('contain', 'Hier können Sie Projekte erstellen.');
+ cy.getByDataCy('portfolio-onboarding-text').should('contain', 'Hier können Sie Projekte erstellen');
cy.getByDataCy('page-footer').should('not.exist');
cy.getByDataCy('create-project-button').should('exist');
});
diff --git a/client/src/components/illustrations/PortfolioIllustration.vue b/client/src/components/illustrations/PortfolioIllustration.vue
index a041a90d..fb20d18c 100644
--- a/client/src/components/illustrations/PortfolioIllustration.vue
+++ b/client/src/components/illustrations/PortfolioIllustration.vue
@@ -1,234 +1,201 @@
diff --git a/client/src/components/illustrations/RoomsIllustration.vue b/client/src/components/illustrations/RoomsIllustration.vue
index 25c124ed..7f90ba6c 100644
--- a/client/src/components/illustrations/RoomsIllustration.vue
+++ b/client/src/components/illustrations/RoomsIllustration.vue
@@ -1,370 +1,270 @@
diff --git a/client/src/components/portfolio/PortfolioOnboarding.vue b/client/src/components/portfolio/PortfolioOnboarding.vue
new file mode 100644
index 00000000..964ac7c8
--- /dev/null
+++ b/client/src/components/portfolio/PortfolioOnboarding.vue
@@ -0,0 +1,51 @@
+
+
+
Portfolio
+
+
Woran denken Sie gerade?
+
+ Hier können Sie Projekte erstellen, um Ihre Gedanken festzuhalten oder Ihre Arbeit zu dokumentieren.
+
+
+
Projekt erstellen
+
+
+
+
+
+
diff --git a/client/src/components/rooms/RoomsOnboarding.vue b/client/src/components/rooms/RoomsOnboarding.vue
index 13ca43ca..66fbf1f4 100644
--- a/client/src/components/rooms/RoomsOnboarding.vue
+++ b/client/src/components/rooms/RoomsOnboarding.vue
@@ -47,31 +47,18 @@
@import '~styles/helpers';
.rooms-onboarding {
- display: flex;
- width: 100vw;
- max-width: 800px;
- flex-direction: column;
- justify-self: center;
- margin: 0 auto;
- grid-column: 1 / -1;
- align-items: center;
+ @include onboarding-page;
&__heading {
margin-bottom: $large-spacing;
}
&__illustration {
- width: 400px;
- height: 320px;
- flex-shrink: 0;
- margin-bottom: $large-spacing;
+ @include onboarding-illustration;
}
&__text {
- @include regular-text;
- text-align: center;
- max-width: 500px;
- margin-bottom: $large-spacing;
+ @include onboarding-text;
}
}
diff --git a/client/src/graphql/gql/queries/allProjects.gql b/client/src/graphql/gql/queries/allProjects.gql
index 82f30894..b626a7ca 100644
--- a/client/src/graphql/gql/queries/allProjects.gql
+++ b/client/src/graphql/gql/queries/allProjects.gql
@@ -1,10 +1,6 @@
#import "../fragments/projectParts.gql"
query ProjectsQuery {
projects {
- edges {
- node {
- ...ProjectParts
- }
- }
+ ...ProjectParts
}
}
diff --git a/client/src/pages/portfolio/portfolio.vue b/client/src/pages/portfolio/portfolio.vue
index 3f956f83..4e4e410f 100644
--- a/client/src/pages/portfolio/portfolio.vue
+++ b/client/src/pages/portfolio/portfolio.vue
@@ -1,19 +1,23 @@
@@ -23,9 +27,11 @@
import ME_QUERY from '@/graphql/gql/queries/meQuery.gql';
import PROJECTS_QUERY from '@/graphql/gql/queries/allProjects.gql';
+ import PortfolioOnboarding from '@/components/portfolio/PortfolioOnboarding';
export default {
components: {
+ PortfolioOnboarding,
ProjectWidget,
AddProject
},
@@ -33,9 +39,6 @@
apollo: {
projects: {
query: PROJECTS_QUERY,
- update(data) {
- return this.$getRidOfEdges(data).projects;
- },
pollInterval: 5000,
},
me: {
diff --git a/client/src/styles/_mixins.scss b/client/src/styles/_mixins.scss
index 1c1b2480..4b6848b0 100644
--- a/client/src/styles/_mixins.scss
+++ b/client/src/styles/_mixins.scss
@@ -78,6 +78,7 @@
@content
}
}
+
@mixin light-border($border-position) {
border-#{$border-position}: 1px solid $color-silver;
}
@@ -188,3 +189,28 @@
border: 0;
min-height: 110px;
}
+
+@mixin onboarding-illustration {
+ width: 400px;
+ height: 320px;
+ flex-shrink: 0;
+ margin-bottom: $large-spacing;
+}
+
+@mixin onboarding-page {
+ display: flex;
+ width: 100vw;
+ max-width: 800px;
+ flex-direction: column;
+ justify-self: center;
+ margin: 0 auto;
+ grid-column: 1 / -1;
+ align-items: center;
+}
+
+@mixin onboarding-text {
+ @include regular-text;
+ text-align: center;
+ max-width: 500px;
+ margin-bottom: $large-spacing;
+}
From 96ed807b2b6051b7d8403148eef24b644763c480 Mon Sep 17 00:00:00 2001
From: Ramon Wenger
Date: Thu, 2 Sep 2021 09:36:17 +0200
Subject: [PATCH 03/28] Update schema for projects
---
.../frontend/portfolio/new-project.spec.js | 50 +++++++++----------
.../frontend/portfolio/project-entry.spec.js | 45 ++++++++---------
.../src/components/content-forms/TextForm.vue | 12 ++++-
server/portfolio/schema.py | 2 +-
server/schema.graphql | 12 +----
5 files changed, 57 insertions(+), 64 deletions(-)
diff --git a/client/cypress/integration/frontend/portfolio/new-project.spec.js b/client/cypress/integration/frontend/portfolio/new-project.spec.js
index 86c43062..2c7346af 100644
--- a/client/cypress/integration/frontend/portfolio/new-project.spec.js
+++ b/client/cypress/integration/frontend/portfolio/new-project.spec.js
@@ -1,35 +1,35 @@
+import {getMinimalMe} from '../../../support/helpers';
+
describe('New project', () => {
+ const MeQuery = getMinimalMe({isTeacher: false});
+ const schoolClass = MeQuery.me.selectedClass;
+
const operations = {
ProjectsQuery: {
- projects: {
- edges: [
- {
- node: {
- id: 'UHJvamVjdE5vZGU6NjY=',
- title: 'Some random title',
- appearance: 'blue',
- description: 'This description rocks',
- slug: 'some-random-title',
- objectives: 'Git gud',
- final: false,
- student: {
- firstName: 'Rachel',
- lastName: 'Green',
- id: 'VXNlck5vZGU6NQ==',
- avatarUrl: '',
- },
- entriesCount: 0,
- },
+ projects: [
+ {
+ id: 'UHJvamVjdE5vZGU6NjY=',
+ title: 'Some random title',
+ appearance: 'blue',
+ description: 'This description rocks',
+ slug: 'some-random-title',
+ objectives: 'Git gud',
+ final: false,
+ schoolClass,
+ student: {
+ firstName: 'Rachel',
+ lastName: 'Green',
+ id: 'VXNlck5vZGU6NQ==',
+ avatarUrl: '',
},
- ],
- },
- },
- MeQuery: {
- me: {},
+ entriesCount: 0,
+ },
+ ],
},
+ MeQuery,
AddProject: variables => ({
addProject: {
- project: Object.assign({}, variables.input.project),
+ project: Object.assign({}, variables.input.project, {schoolClass}),
errors: null,
__typename: 'AddProjectPayload',
},
diff --git a/client/cypress/integration/frontend/portfolio/project-entry.spec.js b/client/cypress/integration/frontend/portfolio/project-entry.spec.js
index a7ca7a85..98657fee 100644
--- a/client/cypress/integration/frontend/portfolio/project-entry.spec.js
+++ b/client/cypress/integration/frontend/portfolio/project-entry.spec.js
@@ -7,30 +7,26 @@ describe('Project Entry', () => {
},
},
ProjectsQuery: {
- projects: {
- edges: [{
- node: {
- id: 'UHJvamVjdE5vZGU6MzM=',
- title: 'Groot',
- appearance: 'red',
- 'description': 'I am Groot',
- 'slug': 'groot',
- 'objectives': 'Be Groot\nBe awesome',
- 'final': false,
- 'student': {
- 'firstName': 'Rachel',
- 'lastName': 'Green',
- 'id': 'VXNlck5vZGU6NQ==',
- 'avatarUrl': '',
- '__typename': 'UserNode',
- },
- 'entriesCount': 2,
- '__typename': 'ProjectNode',
+ projects: [
+ {
+ id: 'UHJvamVjdE5vZGU6MzM=',
+ title: 'Groot',
+ appearance: 'red',
+ 'description': 'I am Groot',
+ 'slug': 'groot',
+ 'objectives': 'Be Groot\nBe awesome',
+ 'final': false,
+ 'student': {
+ 'firstName': 'Rachel',
+ 'lastName': 'Green',
+ 'id': 'VXNlck5vZGU6NQ==',
+ 'avatarUrl': '',
+ '__typename': 'UserNode',
},
- '__typename': 'ProjectNodeEdge',
- }],
- '__typename': 'ProjectNodeConnection',
- },
+ 'entriesCount': 2,
+ '__typename': 'ProjectNode',
+ },
+ ],
},
ProjectQuery: {
'project': {
@@ -85,8 +81,7 @@ describe('Project Entry', () => {
}),
DeleteProjectEntry: {
deleteProjectEntry: {
- success: true,
- __typename: 'DeleteProjectEntryPayload',
+ success: true
},
},
};
diff --git a/client/src/components/content-forms/TextForm.vue b/client/src/components/content-forms/TextForm.vue
index 16856932..25992aec 100644
--- a/client/src/components/content-forms/TextForm.vue
+++ b/client/src/components/content-forms/TextForm.vue
@@ -11,7 +11,15 @@
diff --git a/client/src/components/portfolio/ProjectForm.vue b/client/src/components/portfolio/ProjectForm.vue
index 72d526fc..62bfcf2b 100644
--- a/client/src/components/portfolio/ProjectForm.vue
+++ b/client/src/components/portfolio/ProjectForm.vue
@@ -9,10 +9,6 @@
v-model="localProject.description"
label="Beschreibung"
type="textarea"/>
-
Date: Thu, 2 Sep 2021 09:55:03 +0200
Subject: [PATCH 05/28] Update test
---
.../frontend/portfolio/project-entry.spec.js | 10 +++++-----
client/src/components/portfolio/ProjectEntryForm.vue | 1 +
2 files changed, 6 insertions(+), 5 deletions(-)
diff --git a/client/cypress/integration/frontend/portfolio/project-entry.spec.js b/client/cypress/integration/frontend/portfolio/project-entry.spec.js
index 98657fee..05933c30 100644
--- a/client/cypress/integration/frontend/portfolio/project-entry.spec.js
+++ b/client/cypress/integration/frontend/portfolio/project-entry.spec.js
@@ -81,7 +81,7 @@ describe('Project Entry', () => {
}),
DeleteProjectEntry: {
deleteProjectEntry: {
- success: true
+ success: true,
},
},
};
@@ -113,7 +113,7 @@ describe('Project Entry', () => {
// cy.get('[data-cy=text-form-input]').type('Stay with Rocket\nMeet Quill');
// });
cy.getByDataCy('modal-title').should('contain', 'Beitrag erfassen');
- cy.getByDataCy('project-entry-input').should('exist');
+ cy.getByDataCy('text-form-input').should('exist');
cy.getByDataCy('use-template-button').should('exist').click();
cy.getByDataCy('upload-document-button').should('exist');
cy.getByDataCy('modal-save-button').click();
@@ -130,9 +130,9 @@ describe('Project Entry', () => {
cy.get('[data-cy=project-entry-more]').click();
cy.get('[data-cy=edit-project-entry]').click();
});
- cy.get('[data-cy=activity-input]').within(() => {
- cy.get('[data-cy=text-form-input]').clear().type('Defeat Thanos');
- });
+ cy.getByDataCy('activity-input').should('not.exist');
+
+ cy.getByDataCy('text-form-input').clear().type('Defeat Thanos');
cy.get('[data-cy=modal-save-button]').click();
cy.get('.project-entry__paragraph:first-of-type').contains('Defeat Thanos');
});
diff --git a/client/src/components/portfolio/ProjectEntryForm.vue b/client/src/components/portfolio/ProjectEntryForm.vue
index aba739a6..a92f4b7e 100644
--- a/client/src/components/portfolio/ProjectEntryForm.vue
+++ b/client/src/components/portfolio/ProjectEntryForm.vue
@@ -2,6 +2,7 @@
Beitrag erfassen
From 3951b3f90c88655f4b0cda2802b79723ecb107c4 Mon Sep 17 00:00:00 2001
From: Ramon Wenger
Date: Mon, 20 Sep 2021 20:33:59 +0200
Subject: [PATCH 06/28] Fix unit test
---
server/portfolio/tests/test_project_query.py | 74 +++++---------------
1 file changed, 19 insertions(+), 55 deletions(-)
diff --git a/server/portfolio/tests/test_project_query.py b/server/portfolio/tests/test_project_query.py
index 4ff1d7a4..18f637ad 100644
--- a/server/portfolio/tests/test_project_query.py
+++ b/server/portfolio/tests/test_project_query.py
@@ -3,6 +3,7 @@ from graphene.test import Client
from graphql_relay import to_global_id
from api.schema import schema
+from core.tests.base_test import SkillboxTestCase
from portfolio.factories import ProjectFactory
from portfolio.models import Project
from rooms.models import Room
@@ -11,27 +12,17 @@ from users.models import User, SchoolClass
from users.services import create_users
-class ProjectQuery(TestCase):
+class ProjectQuery(SkillboxTestCase):
def setUp(self):
- create_users()
- self.teacher = User.objects.get(username='teacher')
- self.teacher2 = User.objects.get(username='teacher2')
- self.student = User.objects.get(username='student1')
- self.student2 = User.objects.get(username='student2')
- school_class1 = SchoolClassFactory(users=[self.teacher, self.student])
+ self.createDefault()
+ school_class1 = SchoolClassFactory(users=[self.teacher, self.student1])
school_class2 = SchoolClassFactory(users=[self.teacher2, self.student2])
- self.project1 = ProjectFactory(student=self.student)
+
+ self.project1 = ProjectFactory(student=self.student1)
self.query = '''
query ProjectsQuery {
projects {
- edges {
- node {
- ...ProjectParts
- __typename
- }
- __typename
- }
- __typename
+ ...ProjectParts
}
}
@@ -49,72 +40,45 @@ class ProjectQuery(TestCase):
def test_should_see_own_projects(self):
self.assertEqual(Project.objects.count(), 1)
- request = RequestFactory().get('/')
- request.user = self.student
- self.client = Client(schema=schema, context_value=request)
- result = self.client.execute(self.query)
+ result = self.get_client(self.student1).execute(self.query)
self.assertIsNone(result.get('errors'))
- self.assertEqual(result.get('data').get('projects').get('edges')[0].get('node').get('title'), self.project1.title)
+ self.assertEqual(result.get('data').get('projects')[0].get('title'), self.project1.title)
def test_should_not_see_other_projects(self):
self.assertEqual(Project.objects.count(), 1)
- request = RequestFactory().get('/')
- request.user = self.student2
- self.client = Client(schema=schema, context_value=request)
- result = self.client.execute(self.query)
+ result = self.get_client(self.student2).execute(self.query)
self.assertIsNone(result.get('errors'))
- self.assertEqual(len(result.get('data').get('projects').get('edges')), 0)
+ self.assertEqual(len(result.get('data').get('projects')), 0)
def test_teacher_should_not_see_unfinished_projects(self):
- request = RequestFactory().get('/')
- request.user = self.teacher
- self.client = Client(schema=schema, context_value=request)
-
- result = self.client.execute(self.query)
+ result = self.get_client().execute(self.query)
self.assertIsNone(result.get('errors'))
- self.assertEqual(len(result.get('data').get('projects').get('edges')), 0)
-
- def test_teacher_should_only_see_finished_projects(self):
- self.project1.final = True
- self.assertEqual(Project.objects.count(), 1)
- request = RequestFactory().get('/')
- request.user = self.teacher
- self.client = Client(schema=schema, context_value=request)
-
- result = self.client.execute(self.query)
-
- self.assertIsNone(result.get('errors'))
- self.assertEqual(result.get('data').get('projects').get('edges')[0].get('node').get('title'),
- self.project1.title)
+ self.assertEqual(len(result.get('data').get('projects')), 0)
def test_teacher_should_only_see_finished_projects(self):
self.project1.final = True
self.project1.save()
self.assertEqual(Project.objects.count(), 1)
- request = RequestFactory().get('/')
- request.user = self.teacher
- self.client = Client(schema=schema, context_value=request)
- result = self.client.execute(self.query)
+ result = self.get_client().execute(self.query)
self.assertIsNone(result.get('errors'))
- self.assertEqual(result.get('data').get('projects').get('edges')[0].get('node').get('title'),
+ self.assertEqual(result.get('data').get('projects')[0].get('title'),
self.project1.title)
+
def test_other_teacher_should_not_see_projects(self):
self.project1.final = True
self.project1.save()
self.assertEqual(Project.objects.count(), 1)
- request = RequestFactory().get('/')
- request.user = self.teacher2
- self.client = Client(schema=schema, context_value=request)
- result = self.client.execute(self.query)
+
+ result = self.get_client(self.teacher2).execute(self.query)
self.assertIsNone(result.get('errors'))
- self.assertEqual(len(result.get('data').get('projects').get('edges')), 0)
+ self.assertEqual(len(result.get('data').get('projects')), 0)
From 8e4cffd28f80c73cc5681ee3ccd023d8368e50b7 Mon Sep 17 00:00:00 2001
From: Ramon Wenger
Date: Mon, 20 Sep 2021 21:02:27 +0200
Subject: [PATCH 07/28] Fix some cypress tests
---
client/cypress/fixtures/mocks.js | 2 +-
.../frontend/read-only/portfolio.spec.js | 22 ++++++++-----------
2 files changed, 10 insertions(+), 14 deletions(-)
diff --git a/client/cypress/fixtures/mocks.js b/client/cypress/fixtures/mocks.js
index 66338361..f66111a1 100644
--- a/client/cypress/fixtures/mocks.js
+++ b/client/cypress/fixtures/mocks.js
@@ -1,5 +1,5 @@
const selectedClass = {
- id: 'selectedClassId',
+ id: btoa('SchoolClassNode:selectedClassId'),
name: 'Moordale',
readOnly: false,
code: 'XXXX',
diff --git a/client/cypress/integration/frontend/read-only/portfolio.spec.js b/client/cypress/integration/frontend/read-only/portfolio.spec.js
index 0d0abb44..6f63cff4 100644
--- a/client/cypress/integration/frontend/read-only/portfolio.spec.js
+++ b/client/cypress/integration/frontend/read-only/portfolio.spec.js
@@ -3,20 +3,16 @@ import {getMinimalMe} from '../../../support/helpers';
const getOperations = ({readOnly = false, classReadOnly = false}) => ({
MeQuery: getMinimalMe({readOnly, classReadOnly}),
ProjectsQuery: {
- projects: {
- edges: [
- {
- node: {
- id: 'projectId',
- final: false,
- student: {
- id: btoa('PrivateUserNode:1'),
- },
- entriesCount: 3,
- },
+ projects: [
+ {
+ id: 'projectId',
+ final: false,
+ student: {
+ id: btoa('PrivateUserNode:1'),
},
- ],
- },
+ entriesCount: 3,
+ },
+ ],
},
});
From 1f18f0feeb4db1e90af02117b740af113cf44e2a Mon Sep 17 00:00:00 2001
From: Ramon Wenger
Date: Mon, 27 Sep 2021 14:19:08 +0200
Subject: [PATCH 08/28] Update project list on portfolio page
---
.../frontend/portfolio/projects-page.spec.js | 32 +++++--
.../src/components/portfolio/AddProject.vue | 2 +-
.../portfolio/CreateProjectButton.vue | 26 ++++++
.../src/components/portfolio/OwnerWidget.vue | 2 +-
.../portfolio/PortfolioOnboarding.vue | 17 +---
.../src/components/portfolio/ProjectList.vue | 40 +++++++++
.../components/portfolio/ProjectListItem.vue | 89 +++++++++++++++++++
.../src/components/rooms/EntryCountWidget.vue | 16 +++-
client/src/mixins/me.js | 3 +
client/src/pages/portfolio/portfolio.vue | 28 +++---
client/src/pages/portfolio/project.vue | 5 +-
client/src/router/portfolio.routes.js | 2 +-
12 files changed, 216 insertions(+), 46 deletions(-)
create mode 100644 client/src/components/portfolio/CreateProjectButton.vue
create mode 100644 client/src/components/portfolio/ProjectList.vue
create mode 100644 client/src/components/portfolio/ProjectListItem.vue
diff --git a/client/cypress/integration/frontend/portfolio/projects-page.spec.js b/client/cypress/integration/frontend/portfolio/projects-page.spec.js
index 4c580227..4de3df5d 100644
--- a/client/cypress/integration/frontend/portfolio/projects-page.spec.js
+++ b/client/cypress/integration/frontend/portfolio/projects-page.spec.js
@@ -10,9 +10,7 @@ describe('Projects page', () => {
const operations = {
MeQuery,
ProjectsQuery: {
- projects: {
- edges: [],
- },
+ projects: [],
},
};
cy.mockGraphqlOps({
@@ -24,16 +22,30 @@ describe('Projects page', () => {
cy.getByDataCy('portfolio-onboarding-subtitle').should('contain', 'Woran denken Sie gerade');
cy.getByDataCy('portfolio-onboarding-text').should('contain', 'Hier können Sie Projekte erstellen');
cy.getByDataCy('page-footer').should('not.exist');
- cy.getByDataCy('create-project-button').should('exist');
+ cy.getByDataCy('add-project-button').should('exist');
});
it('displays the project list', () => {
+ const projectTitle = 'What is love?';
+
const operations = {
MeQuery,
ProjectsQuery: {
- projects: {
- edges: [],
- },
+ projects: [
+ {
+ title: projectTitle,
+ student: {
+ firstName: 'Bilbo',
+ lastName: 'Baggins',
+ avatarUrl: ''
+ },
+ entriesCount: 0,
+ entries: {
+ nodes: [],
+
+ }
+ }
+ ],
},
};
@@ -42,9 +54,11 @@ describe('Projects page', () => {
});
cy.visit('/portfolio');
cy.getByDataCy('page-title').should('contain', 'Portfolio');
- cy.getByDataCy('create-project-button').should('exist');
+ cy.getByDataCy('add-project-button').should('exist');
cy.getByDataCy('project-list').should('exist');
cy.getByDataCy('project').should('have.length', 1);
- cy.getByDataCy('project-owner').should('contain', 'Bilbo Baggins');
+ cy.getByDataCy('owner-name').should('contain', 'Bilbo Baggins');
+ cy.getByDataCy('project-title').should('contain', projectTitle);
+ cy.getByDataCy('entry-count').should('contain', 0);
});
});
diff --git a/client/src/components/portfolio/AddProject.vue b/client/src/components/portfolio/AddProject.vue
index 66fb4d8b..95e33f94 100644
--- a/client/src/components/portfolio/AddProject.vue
+++ b/client/src/components/portfolio/AddProject.vue
@@ -1,7 +1,7 @@
+ data-cy="add-project-widget"/>
+
+
diff --git a/client/src/components/portfolio/OwnerWidget.vue b/client/src/components/portfolio/OwnerWidget.vue
index aecf5fe0..8970a9c2 100644
--- a/client/src/components/portfolio/OwnerWidget.vue
+++ b/client/src/components/portfolio/OwnerWidget.vue
@@ -3,7 +3,7 @@
-
+
{{ name }}
diff --git a/client/src/components/portfolio/PortfolioOnboarding.vue b/client/src/components/portfolio/PortfolioOnboarding.vue
index 33567d58..76d9b9b7 100644
--- a/client/src/components/portfolio/PortfolioOnboarding.vue
+++ b/client/src/components/portfolio/PortfolioOnboarding.vue
@@ -15,26 +15,15 @@
Hier können Sie Projekte erstellen, um Ihre Gedanken festzuhalten oder Ihre Arbeit zu dokumentieren.
- Projekt erstellen
+
diff --git a/client/src/components/portfolio/ProjectList.vue b/client/src/components/portfolio/ProjectList.vue
new file mode 100644
index 00000000..742aac6d
--- /dev/null
+++ b/client/src/components/portfolio/ProjectList.vue
@@ -0,0 +1,40 @@
+
+
+
+
+
+
+
diff --git a/client/src/components/portfolio/ProjectListItem.vue b/client/src/components/portfolio/ProjectListItem.vue
new file mode 100644
index 00000000..c80641e2
--- /dev/null
+++ b/client/src/components/portfolio/ProjectListItem.vue
@@ -0,0 +1,89 @@
+
+
+
+ {{ project.title }}
+
+
+
+
+
+
+
+
+
+
+
+
+
diff --git a/client/src/components/rooms/EntryCountWidget.vue b/client/src/components/rooms/EntryCountWidget.vue
index 934b9899..c032e2bd 100644
--- a/client/src/components/rooms/EntryCountWidget.vue
+++ b/client/src/components/rooms/EntryCountWidget.vue
@@ -1,7 +1,7 @@
- {{ entryCount }} {{ entryCount === 1 ? 'Beitrag' : 'Beiträge' }}
+ {{ entryCount }} {{ entryCount === 1 ? 'Beitrag' : 'Beiträge' }}
@@ -9,11 +9,19 @@
import Cards from '@/components/icons/Cards.vue';
export default {
- props: ['entryCount'],
+ props: {
+ entryCount: {
+ type: Number,
+ },
+ verbose: {
+ type: Boolean,
+ default: true,
+ },
+ },
components: {
- Cards
- }
+ Cards,
+ },
};
diff --git a/client/src/mixins/me.js b/client/src/mixins/me.js
index 87284aec..04203cf3 100644
--- a/client/src/mixins/me.js
+++ b/client/src/mixins/me.js
@@ -34,6 +34,9 @@ export default {
canManageContent() {
return this.me.isTeacher;
},
+ isReadOnly() {
+ return this.me.readOnly || this.me.selectedClass.readOnly;
+ },
currentClassName() {
let currentClass = this.me.schoolClasses.find(schoolClass => {
return schoolClass.id === this.me.selectedClass.id;
diff --git a/client/src/pages/portfolio/portfolio.vue b/client/src/pages/portfolio/portfolio.vue
index 4e4e410f..184a01ec 100644
--- a/client/src/pages/portfolio/portfolio.vue
+++ b/client/src/pages/portfolio/portfolio.vue
@@ -2,18 +2,12 @@
@@ -28,9 +22,13 @@
import ME_QUERY from '@/graphql/gql/queries/meQuery.gql';
import PROJECTS_QUERY from '@/graphql/gql/queries/allProjects.gql';
import PortfolioOnboarding from '@/components/portfolio/PortfolioOnboarding';
+ import CreateProjectButton from '@/components/portfolio/CreateProjectButton';
+ import ProjectList from '@/components/portfolio/ProjectList';
export default {
components: {
+ ProjectList,
+ CreateProjectButton,
PortfolioOnboarding,
ProjectWidget,
AddProject
@@ -70,15 +68,21 @@
@supports (display: grid) {
display: grid;
grid-template-columns: minmax(min-content, 840px);
+ grid-template-rows: auto;
}
grid-row-gap: 30px;
- grid-auto-rows: 225px;
+ grid-auto-rows: auto;
max-width: 840px;
width: 100vw;
/*justify-self: center;*/
box-sizing: border-box;
padding: $large-spacing $medium-spacing;
+ &__create-button {
+ display: inline-flex;
+ width: 150px;
+ }
+
&__page {
display: flex;
flex-direction: column;
diff --git a/client/src/pages/portfolio/project.vue b/client/src/pages/portfolio/project.vue
index 0220d58e..b46305ac 100644
--- a/client/src/pages/portfolio/project.vue
+++ b/client/src/pages/portfolio/project.vue
@@ -123,12 +123,9 @@
diff --git a/client/src/components/portfolio/NewProjectEntryWizard.vue b/client/src/components/portfolio/NewProjectEntryWizard.vue
index 53931b18..c73f80e0 100644
--- a/client/src/components/portfolio/NewProjectEntryWizard.vue
+++ b/client/src/components/portfolio/NewProjectEntryWizard.vue
@@ -19,7 +19,7 @@
data() {
return {
projectEntry: {
- content: '',
+ description: '',
documentUrl: ''
}
};
diff --git a/client/src/components/portfolio/ProjectEntryForm.vue b/client/src/components/portfolio/ProjectEntryForm.vue
index dce6209c..d21d8c5c 100644
--- a/client/src/components/portfolio/ProjectEntryForm.vue
+++ b/client/src/components/portfolio/ProjectEntryForm.vue
@@ -7,13 +7,28 @@
Beitrag erfassen
'' ? {
- url: this.localProjectEntry.documentUrl,
- } : {};
- },
- content() {
- return {
- text: this.localProjectEntry.content,
- };
- },
- },
-
methods: {
setDocumentUrl(url) {
this.localProjectEntry.documentUrl = url;
},
+ useTemplate() {
+ this.localProjectEntry.description = `${this.localProjectEntry.description}${PROJECT_ENTRY_TEMPLATE}`;
+ },
},
};
@@ -82,10 +98,46 @@
.project-entry-modal {
display: flex;
+ flex-direction: column;
+
+ &__form-field {
+ @include inputstyle;
+ padding: 0;
+ display: flex;
+ flex-direction: column;
+ grid-template-rows: auto 1rem;
+ grid-template-columns: 1fr 1fr;
+ width: 100%;
+ }
+
+ &__textarea {
+ @include auto-grow;
+ border: 0;
+ min-height: 400px;
+ padding: $medium-spacing;
+ }
+
+ &__buttons {
+ display: grid;
+ grid-template-columns: 1fr 1fr;
+ padding: $medium-spacing;
+ }
+
+ &__button {
+ @include regular-text;
+
+ &--template {
+ }
+
+ &--document {
+ }
+ }
&__heading {
@include heading-3;
margin-bottom: 0;
}
+
}
+
diff --git a/client/src/components/ui/ButtonWithIconAndText.vue b/client/src/components/ui/ButtonWithIconAndText.vue
new file mode 100644
index 00000000..5a4cdb1f
--- /dev/null
+++ b/client/src/components/ui/ButtonWithIconAndText.vue
@@ -0,0 +1,51 @@
+
+
+
+ {{ text }}
+
+
+
+
+
+
diff --git a/client/src/components/ui/file-upload/FileUpload.vue b/client/src/components/ui/file-upload/FileUpload.vue
new file mode 100644
index 00000000..09205266
--- /dev/null
+++ b/client/src/components/ui/file-upload/FileUpload.vue
@@ -0,0 +1,45 @@
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
diff --git a/client/src/components/ui/file-upload/SimpleFileUpload.vue b/client/src/components/ui/file-upload/SimpleFileUpload.vue
new file mode 100644
index 00000000..c6a3c8ad
--- /dev/null
+++ b/client/src/components/ui/file-upload/SimpleFileUpload.vue
@@ -0,0 +1,71 @@
+
+
+
+
+
+
+
+
+
+
diff --git a/client/src/components/ui/file-upload/SimpleFileUploadHiddenInput.vue b/client/src/components/ui/file-upload/SimpleFileUploadHiddenInput.vue
new file mode 100644
index 00000000..c8109914
--- /dev/null
+++ b/client/src/components/ui/file-upload/SimpleFileUploadHiddenInput.vue
@@ -0,0 +1,27 @@
+
+
+
+
+
diff --git a/client/src/components/ui/file-upload/SimpleFileUploadIcon.vue b/client/src/components/ui/file-upload/SimpleFileUploadIcon.vue
new file mode 100644
index 00000000..d7ed76c4
--- /dev/null
+++ b/client/src/components/ui/file-upload/SimpleFileUploadIcon.vue
@@ -0,0 +1,24 @@
+
+
+
+
+
+
+
+
+
diff --git a/client/src/components/ui/file-upload/SimpleFileUploadIconAndText.vue b/client/src/components/ui/file-upload/SimpleFileUploadIconAndText.vue
new file mode 100644
index 00000000..7f43c1f9
--- /dev/null
+++ b/client/src/components/ui/file-upload/SimpleFileUploadIconAndText.vue
@@ -0,0 +1,13 @@
+
+
+
+
+
diff --git a/client/src/components/SimpleFileUpload.vue b/client/src/components/ui/file-upload/SimpleFileUploadWithIcon.vue
similarity index 52%
rename from client/src/components/SimpleFileUpload.vue
rename to client/src/components/ui/file-upload/SimpleFileUploadWithIcon.vue
index aa5058db..de4f69e6 100644
--- a/client/src/components/SimpleFileUpload.vue
+++ b/client/src/components/ui/file-upload/SimpleFileUploadWithIcon.vue
@@ -1,41 +1,27 @@
-
-
-
-
+ @click.native="clickUploadCare"/>
+
+
diff --git a/client/src/components/portfolio/EditProject.vue b/client/src/components/portfolio/EditProject.vue
index 44314e9a..cc0ec956 100644
--- a/client/src/components/portfolio/EditProject.vue
+++ b/client/src/components/portfolio/EditProject.vue
@@ -27,7 +27,6 @@
id: project.id,
title: project.title,
description: project.description,
- appearance: project.appearance,
objectives: project.objectives
}
}
diff --git a/client/src/components/portfolio/ProjectForm.vue b/client/src/components/portfolio/ProjectForm.vue
index 62bfcf2b..67da6375 100644
--- a/client/src/components/portfolio/ProjectForm.vue
+++ b/client/src/components/portfolio/ProjectForm.vue
@@ -9,10 +9,6 @@
v-model="localProject.description"
label="Beschreibung"
type="textarea"/>
-
@@ -39,14 +41,13 @@
UserWidget,
Logo,
CurrentClass,
- Hamburger
+ Hamburger,
},
};
From 348e9198b67145e92810144fdc2806a1e2add69d Mon Sep 17 00:00:00 2001
From: Ramon Wenger
Date: Sun, 10 Oct 2021 21:53:54 +0200
Subject: [PATCH 23/28] Add share link to project page
---
.../frontend/portfolio/project-page.spec.js | 1 -
client/src/components/icons/ShareIcon.vue | 10 ++++
.../components/portfolio/ProjectActions.vue | 55 ++++++++-----------
client/src/components/portfolio/ShareLink.vue | 43 +++++++++++++++
client/src/components/ui/WidgetPopover.vue | 8 ++-
.../src/graphql/gql/queries/projectQuery.gql | 1 -
.../src/mixins/update-project-share-state.js | 32 +++++++++++
client/src/pages/portfolio/project.vue | 50 ++++++++++++-----
8 files changed, 150 insertions(+), 50 deletions(-)
create mode 100644 client/src/components/icons/ShareIcon.vue
create mode 100644 client/src/components/portfolio/ShareLink.vue
create mode 100644 client/src/mixins/update-project-share-state.js
diff --git a/client/cypress/integration/frontend/portfolio/project-page.spec.js b/client/cypress/integration/frontend/portfolio/project-page.spec.js
index e3e469e3..fefaff9e 100644
--- a/client/cypress/integration/frontend/portfolio/project-page.spec.js
+++ b/client/cypress/integration/frontend/portfolio/project-page.spec.js
@@ -101,7 +101,6 @@ describe('Project Page', () => {
cy.getByDataCy('project-actions').click();
cy.getByDataCy('delete-project').should('exist');
cy.getByDataCy('edit-project').should('exist');
- cy.getByDataCy('share-project').should('exist');
});
describe('Project Entry', () => {
diff --git a/client/src/components/icons/ShareIcon.vue b/client/src/components/icons/ShareIcon.vue
new file mode 100644
index 00000000..0a04014f
--- /dev/null
+++ b/client/src/components/icons/ShareIcon.vue
@@ -0,0 +1,10 @@
+
+
+
diff --git a/client/src/components/portfolio/ProjectActions.vue b/client/src/components/portfolio/ProjectActions.vue
index a515cd2b..62a16ca2 100644
--- a/client/src/components/portfolio/ProjectActions.vue
+++ b/client/src/components/portfolio/ProjectActions.vue
@@ -23,14 +23,14 @@
Projekt teilen
+ @click="updateProjectShareState(id, true)">Projekt teilen
Projekt nicht mehr teilen
+ @click="updateProjectShareState(id, false)">Projekt nicht mehr teilen
@@ -42,12 +42,27 @@ import Ellipses from '@/components/icons/Ellipses.vue';
import WidgetPopover from '@/components/ui/WidgetPopover';
import DELETE_PROJECT_MUTATION from '@/graphql/gql/mutations/deleteProject.gql';
-import PROJECT_QUERY from '@/graphql/gql/queries/projectQuery.gql';
import PROJECTS_QUERY from '@/graphql/gql/queries/allProjects.gql';
-import UPDATE_PROJECT_SHARED_STATE_MUTATION from '@/graphql/gql/mutations/updateProjectSharedState.gql';
+
+import updateProjectShareState from '@/mixins/update-project-share-state';
export default {
- props: ['id', 'final'],
+ props: {
+ id: {
+ type: String,
+ default: '',
+ },
+ final: {
+ type: Boolean,
+ default: false,
+ },
+ shareButtons: {
+ type: Boolean,
+ default: true,
+ },
+ },
+
+ mixins: [updateProjectShareState],
components: {
Ellipses,
@@ -89,32 +104,6 @@ export default {
this.$router.push('/portfolio');
});
},
- updateShareState(id, state) {
- const project = this;
- this.$apollo.mutate({
- mutation: UPDATE_PROJECT_SHARED_STATE_MUTATION,
- variables: {
- input: {
- id: this.id,
- shared: state,
- },
- },
- update(store, {data: {updateProjectSharedState: {shared, errors}}}) {
- if (!errors) {
- const query = PROJECT_QUERY;
- const variables = {
- id: project.id,
- };
- const data = store.readQuery({query, variables});
-
- if (data) {
- data.project.final = shared;
- store.writeQuery({query, variables, data});
- }
- }
- },
- });
- },
},
};
diff --git a/client/src/components/portfolio/ShareLink.vue b/client/src/components/portfolio/ShareLink.vue
new file mode 100644
index 00000000..124f1c0b
--- /dev/null
+++ b/client/src/components/portfolio/ShareLink.vue
@@ -0,0 +1,43 @@
+
+
+
+
+ Mit Lehrperson teilen
+ Nicht mehr teilen
+
+
+
+
+
+
+
diff --git a/client/src/components/ui/WidgetPopover.vue b/client/src/components/ui/WidgetPopover.vue
index 340a5beb..93e3a1c0 100644
--- a/client/src/components/ui/WidgetPopover.vue
+++ b/client/src/components/ui/WidgetPopover.vue
@@ -11,7 +11,13 @@