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