From e1211e0a84f33c8bdedc0083415fcc6ab7207c59 Mon Sep 17 00:00:00 2001
From: robban64 <carl@schonfelder.se>
Date: Tue, 27 Apr 2021 16:24:53 +0200
Subject: [PATCH] fix: return media object instead of media_id

---
 client/src/actions/presentation.test.ts         |  2 +-
 client/src/interfaces/ApiModels.ts              |  8 ++++----
 client/src/interfaces/ApiRichModels.ts          |  3 ++-
 .../components/ImageComponentDisplay.test.tsx   | 10 +++++++++-
 .../components/ImageComponentDisplay.tsx        |  2 +-
 .../slideSettingsComponents/Images.tsx          |  6 +++---
 client/src/reducers/editorReducer.ts            |  1 +
 client/src/reducers/presentationReducer.ts      |  2 ++
 server/app/apis/answers.py                      |  4 ++--
 server/app/apis/competitions.py                 |  5 ++++-
 server/app/apis/slides.py                       |  1 +
 server/app/core/rich_schemas.py                 |  6 ++----
 server/app/core/schemas.py                      | 17 ++++-------------
 server/app/database/controller/add.py           |  4 ++--
 server/app/database/controller/copy.py          |  4 ++--
 server/app/database/models.py                   |  3 +--
 server/configmodule.py                          |  2 +-
 17 files changed, 42 insertions(+), 38 deletions(-)

diff --git a/client/src/actions/presentation.test.ts b/client/src/actions/presentation.test.ts
index 59d4c967..d95d9db7 100644
--- a/client/src/actions/presentation.test.ts
+++ b/client/src/actions/presentation.test.ts
@@ -26,7 +26,7 @@ it('dispatches no actions when failing to get competitions', async () => {
 })
 
 it('dispatches correct actions when setting slide', () => {
-  const testSlide: Slide = { competition_id: 0, id: 5, order: 5, timer: 20, title: '' }
+  const testSlide: Slide = { competition_id: 0, id: 5, order: 5, timer: 20, title: '', background_image_id: 0 }
   const expectedActions = [{ type: Types.SET_PRESENTATION_SLIDE, payload: testSlide }]
   const store = mockStore({})
   setCurrentSlide(testSlide)(store.dispatch)
diff --git a/client/src/interfaces/ApiModels.ts b/client/src/interfaces/ApiModels.ts
index b3cede4d..4de675fb 100644
--- a/client/src/interfaces/ApiModels.ts
+++ b/client/src/interfaces/ApiModels.ts
@@ -38,13 +38,14 @@ export interface Slide {
   order: number
   timer: number
   title: string
+  background_image?: Media
 }
 
 export interface Competition extends NameID {
   font: string
   city_id: number
   year: number
-  background_image_id: number
+  background_image?: Media
 }
 
 export interface Team extends NameID {
@@ -69,7 +70,7 @@ export interface QuestionAnswer {
   id: number
   question_id: number
   team_id: number
-  data: string
+  answer: string
   score: number
 }
 
@@ -83,8 +84,7 @@ export interface Component {
 }
 
 export interface ImageComponent extends Component {
-  media_id: number
-  filename: string
+  media: Media
 }
 
 export interface TextComponent extends Component {
diff --git a/client/src/interfaces/ApiRichModels.ts b/client/src/interfaces/ApiRichModels.ts
index 2388f830..b9ca9f33 100644
--- a/client/src/interfaces/ApiRichModels.ts
+++ b/client/src/interfaces/ApiRichModels.ts
@@ -7,6 +7,7 @@ export interface RichCompetition {
   city_id: number
   slides: RichSlide[]
   teams: RichTeam[]
+  background_image?: Media
 }
 
 export interface RichSlide {
@@ -15,9 +16,9 @@ export interface RichSlide {
   timer: number
   title: string
   competition_id: number
+  background_image?: Media
   components: Component[]
   questions: RichQuestion[]
-  medias: Media[]
 }
 
 export interface RichTeam {
diff --git a/client/src/pages/presentationEditor/components/ImageComponentDisplay.test.tsx b/client/src/pages/presentationEditor/components/ImageComponentDisplay.test.tsx
index 9e78f8e1..eb823693 100644
--- a/client/src/pages/presentationEditor/components/ImageComponentDisplay.test.tsx
+++ b/client/src/pages/presentationEditor/components/ImageComponentDisplay.test.tsx
@@ -5,7 +5,15 @@ import ImageComponentDisplay from './ImageComponentDisplay'
 it('renders competition settings', () => {
   render(
     <ImageComponentDisplay
-      component={{ id: 0, x: 0, y: 0, w: 0, h: 0, data: { media_id: 0, filename: '' }, type_id: 2 }}
+      component={{
+        id: 0,
+        x: 0,
+        y: 0,
+        w: 0,
+        h: 0,
+        media: { id: 0, mediatype_id: 0, user_id: 0, filename: '' },
+        type_id: 2,
+      }}
       width={0}
       height={0}
     />
diff --git a/client/src/pages/presentationEditor/components/ImageComponentDisplay.tsx b/client/src/pages/presentationEditor/components/ImageComponentDisplay.tsx
index 5e409d57..e783bdb2 100644
--- a/client/src/pages/presentationEditor/components/ImageComponentDisplay.tsx
+++ b/client/src/pages/presentationEditor/components/ImageComponentDisplay.tsx
@@ -10,7 +10,7 @@ type ImageComponentProps = {
 const ImageComponentDisplay = ({ component, width, height }: ImageComponentProps) => {
   return (
     <img
-      src={`http://localhost:5000/static/images/${component.filename}`}
+      src={`http://localhost:5000/static/images/${component.media?.filename}`}
       height={height}
       width={width}
       draggable={false}
diff --git a/client/src/pages/presentationEditor/components/slideSettingsComponents/Images.tsx b/client/src/pages/presentationEditor/components/slideSettingsComponents/Images.tsx
index 0a0d1233..d8cc164c 100644
--- a/client/src/pages/presentationEditor/components/slideSettingsComponents/Images.tsx
+++ b/client/src/pages/presentationEditor/components/slideSettingsComponents/Images.tsx
@@ -65,7 +65,7 @@ const Images = ({ activeSlide, competitionId }: ImagesProps) => {
   const handleCloseimageClick = async (image: ImageComponent) => {
     // Removes selected image component and deletes its file from the server.
     await axios
-      .delete(`/api/media/images/${image.media_id}`)
+      .delete(`/api/media/images/${image.media?.id}`)
       .then(() => {
         dispatch(getEditorCompetition(competitionId))
       })
@@ -98,9 +98,9 @@ const Images = ({ activeSlide, competitionId }: ImagesProps) => {
         images.map((image) => (
           <div key={image.id}>
             <ListItem divider button>
-              <ImportedImage src={`http://localhost:5000/static/images/thumbnail_${image.filename}`} />
+              <ImportedImage src={`http://localhost:5000/static/images/thumbnail_${image.media?.filename}`} />
               <Center>
-                <ListItemText primary={image.filename} />
+                <ListItemText primary={image.media?.filename} />
               </Center>
               <CloseIcon onClick={() => handleCloseimageClick(image)} />
             </ListItem>
diff --git a/client/src/reducers/editorReducer.ts b/client/src/reducers/editorReducer.ts
index 20d0b428..88e6e368 100644
--- a/client/src/reducers/editorReducer.ts
+++ b/client/src/reducers/editorReducer.ts
@@ -16,6 +16,7 @@ const initialState: EditorState = {
     city_id: 1,
     slides: [],
     teams: [],
+    background_image: undefined,
   },
   activeSlideId: -1,
   loading: true,
diff --git a/client/src/reducers/presentationReducer.ts b/client/src/reducers/presentationReducer.ts
index 0b16c023..b657f5a4 100644
--- a/client/src/reducers/presentationReducer.ts
+++ b/client/src/reducers/presentationReducer.ts
@@ -19,6 +19,7 @@ const initialState: PresentationState = {
     slides: [],
     year: 0,
     teams: [],
+    background_image: undefined,
   },
   slide: {
     competition_id: 0,
@@ -26,6 +27,7 @@ const initialState: PresentationState = {
     order: 0,
     timer: 0,
     title: '',
+    background_image: undefined,
   },
   code: '',
   timer: {
diff --git a/server/app/apis/answers.py b/server/app/apis/answers.py
index 6bb30ddb..e3e225b9 100644
--- a/server/app/apis/answers.py
+++ b/server/app/apis/answers.py
@@ -10,12 +10,12 @@ schema = QuestionAnswerDTO.schema
 list_schema = QuestionAnswerDTO.list_schema
 
 question_answer_parser = reqparse.RequestParser()
-question_answer_parser.add_argument("data", type=dict, required=True, location="json")
+question_answer_parser.add_argument("answer", type=str, required=True, location="json")
 question_answer_parser.add_argument("score", type=int, required=True, location="json")
 question_answer_parser.add_argument("question_id", type=int, required=True, location="json")
 
 question_answer_edit_parser = reqparse.RequestParser()
-question_answer_edit_parser.add_argument("data", type=dict, default=None, location="json")
+question_answer_edit_parser.add_argument("answer", type=str, default=None, location="json")
 question_answer_edit_parser.add_argument("score", type=int, default=None, location="json")
 
 
diff --git a/server/app/apis/competitions.py b/server/app/apis/competitions.py
index e806f34a..fded52ec 100644
--- a/server/app/apis/competitions.py
+++ b/server/app/apis/competitions.py
@@ -18,6 +18,9 @@ competition_parser.add_argument("name", type=str, location="json")
 competition_parser.add_argument("year", type=int, location="json")
 competition_parser.add_argument("city_id", type=int, location="json")
 
+competition_edit_parser = competition_parser.copy()
+competition_edit_parser.add_argument("background_image_id", default=None, type=int, location="json")
+
 competition_search_parser = search_parser.copy()
 competition_search_parser.add_argument("name", type=str, default=None, location="args")
 competition_search_parser.add_argument("year", type=int, default=None, location="args")
@@ -48,7 +51,7 @@ class Competitions(Resource):
 
     @check_jwt(editor=True)
     def put(self, competition_id):
-        args = competition_parser.parse_args(strict=True)
+        args = competition_edit_parser.parse_args(strict=True)
         item = dbc.get.one(Competition, competition_id)
         item = dbc.edit.default(item, **args)
 
diff --git a/server/app/apis/slides.py b/server/app/apis/slides.py
index 1823531e..da29633a 100644
--- a/server/app/apis/slides.py
+++ b/server/app/apis/slides.py
@@ -13,6 +13,7 @@ slide_parser = reqparse.RequestParser()
 slide_parser.add_argument("order", type=int, default=None, location="json")
 slide_parser.add_argument("title", type=str, default=None, location="json")
 slide_parser.add_argument("timer", type=int, default=None, location="json")
+slide_parser.add_argument("background_image_id", default=None, type=int, location="json")
 
 
 @api.route("")
diff --git a/server/app/core/rich_schemas.py b/server/app/core/rich_schemas.py
index da5f58d6..ae1c6ab6 100644
--- a/server/app/core/rich_schemas.py
+++ b/server/app/core/rich_schemas.py
@@ -42,8 +42,7 @@ class SlideSchemaRich(RichSchema):
     title = ma.auto_field()
     timer = ma.auto_field()
     competition_id = ma.auto_field()
-    background_image_id = ma.auto_field()
-    background_image = ma.Function(lambda x: x.background_image.filename if x.background_image is not None else "")
+    background_image = fields.Nested(schemas.MediaSchema, many=False)
     questions = fields.Nested(QuestionSchemaRich, many=True)
     components = fields.Nested(schemas.ComponentSchema, many=True)
 
@@ -56,8 +55,7 @@ class CompetitionSchemaRich(RichSchema):
     name = ma.auto_field()
     year = ma.auto_field()
     city_id = ma.auto_field()
-    background_image_id = ma.auto_field()
-    background_image = ma.Function(lambda x: x.background_image.filename if x.background_image is not None else "")
+    background_image = fields.Nested(schemas.MediaSchema, many=False)
 
     slides = fields.Nested(
         SlideSchemaRich,
diff --git a/server/app/core/schemas.py b/server/app/core/schemas.py
index 1704b651..0abe0252 100644
--- a/server/app/core/schemas.py
+++ b/server/app/core/schemas.py
@@ -65,7 +65,7 @@ class QuestionAnswerSchema(BaseSchema):
         model = models.QuestionAnswer
 
     id = ma.auto_field()
-    data = ma.Function(lambda obj: obj.data)
+    answer = ma.auto_field()
     score = ma.auto_field()
     question_id = ma.auto_field()
     team_id = ma.auto_field()
@@ -116,8 +116,7 @@ class SlideSchema(BaseSchema):
     title = ma.auto_field()
     timer = ma.auto_field()
     competition_id = ma.auto_field()
-    background_image_id = ma.auto_field()
-    background_image = ma.Function(lambda x: x.background_image.filename if x.background_image is not None else "")
+    background_image = fields.Nested(MediaSchema, many=False)
 
 
 class TeamSchema(BaseSchema):
@@ -148,20 +147,13 @@ class CompetitionSchema(BaseSchema):
     name = ma.auto_field()
     year = ma.auto_field()
     city_id = ma.auto_field()
-    background_image_id = ma.auto_field()
-    background_image = ma.Function(lambda x: x.background_image.filename if x.background_image is not None else "")
+    background_image = fields.Nested(MediaSchema, many=False)
 
 
 class ComponentSchema(BaseSchema):
     class Meta(BaseSchema.Meta):
         model = models.Component
 
-    @post_dump
-    def handle_filename(self, data, *args, **kwargs):
-        if data["filename"] == "":
-            del data["filename"]
-        return data
-
     id = ma.auto_field()
     x = ma.auto_field()
     y = ma.auto_field()
@@ -171,5 +163,4 @@ class ComponentSchema(BaseSchema):
     type_id = ma.auto_field()
 
     text = fields.fields.String()
-    media_id = fields.fields.Integer()
-    filename = ma.Function(lambda x: x.media.filename if hasattr(x, "media_id") else "")
+    media = fields.Nested(MediaSchema, many=False)
diff --git a/server/app/database/controller/add.py b/server/app/database/controller/add.py
index dceccfe8..a83f5b95 100644
--- a/server/app/database/controller/add.py
+++ b/server/app/database/controller/add.py
@@ -246,5 +246,5 @@ def question_alternative(text, value, question_id):
     return db_add(QuestionAlternative(text, value, question_id))
 
 
-def question_answer(data, score, question_id, team_id):
-    return db_add(QuestionAnswer(data, score, question_id, team_id))
+def question_answer(answer, score, question_id, team_id):
+    return db_add(QuestionAnswer(answer, score, question_id, team_id))
diff --git a/server/app/database/controller/copy.py b/server/app/database/controller/copy.py
index 87e3f831..7e9d28d0 100644
--- a/server/app/database/controller/copy.py
+++ b/server/app/database/controller/copy.py
@@ -77,8 +77,7 @@ def slide_to_competition(item_slide_old, item_competition):
     item_slide_new.body = item_slide_old.body
     item_slide_new.timer = item_slide_old.timer
     item_slide_new.settings = item_slide_old.settings
-
-    # TODO: Add background image
+    item_slide_new.background_image_id = item_slide_old.background_image_id
 
     for item_component in item_slide_old.components:
         _component(item_component, item_slide_new)
@@ -109,6 +108,7 @@ def competition(item_competition_old):
         item_competition_old.font,
     )
     # TODO: Add background image
+    item_competition_new.background_image_id = item_competition_old.background_image_id
 
     for item_slide in item_competition_old.slides:
         slide_to_competition(item_slide, item_competition_new)
diff --git a/server/app/database/models.py b/server/app/database/models.py
index 1ed99d91..30129389 100644
--- a/server/app/database/models.py
+++ b/server/app/database/models.py
@@ -1,5 +1,4 @@
 from app.core import bcrypt, db
-from app.database import Dictionary
 from sqlalchemy.ext.hybrid import hybrid_method, hybrid_property
 
 from app.database.types import ID_IMAGE_COMPONENT, ID_TEXT_COMPONENT
@@ -175,7 +174,7 @@ class QuestionAlternative(db.Model):
 class QuestionAnswer(db.Model):
     __table_args__ = (db.UniqueConstraint("question_id", "team_id"),)
     id = db.Column(db.Integer, primary_key=True)
-    data = db.Column(Dictionary(), nullable=False)
+    answer = db.Column(db.String(STRING_SIZE), nullable=False)
     score = db.Column(db.Integer, nullable=False, default=0)
 
     question_id = db.Column(db.Integer, db.ForeignKey("question.id"), nullable=False)
diff --git a/server/configmodule.py b/server/configmodule.py
index 9e78490d..8c07211e 100644
--- a/server/configmodule.py
+++ b/server/configmodule.py
@@ -20,7 +20,7 @@ class Config:
 
 
 class DevelopmentConfig(Config):
-    DEBUG = False
+    DEBUG = True
     SQLALCHEMY_ECHO = False
     # HOST = "localhost"
     # PORT = 5432
-- 
GitLab