diff --git a/client/src/actions/presentation.test.ts b/client/src/actions/presentation.test.ts index 59d4c967254a483b7015eb6c78413d009b9604b4..d95d9db767b214e4e2583a1a903ed06b226eec16 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 b3cede4dd858c725baceceab6a31c77792cf24d1..4de675fb43475f432cfa2f13a93f30c8ef46e782 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 2388f8308ba8bfa38d4d5376040f937244f4cbd5..b9ca9f333640071518a9f216e08a77af670eb153 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 9e78f8e13fd94cf36434ef63ed7504695208de1c..eb823693c7f192489f4e0cc8e15f5f1af3f6a470 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 5e409d57f0e6f8a7503745760dd9078440fa732c..e783bdb22e743c345611b92bfc11d161fc2fe957 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 0a0d12337cb54da77f4855cfdc00d02bf9a43f59..d8cc164cc8de656a7f732a92d35f9eb45572d453 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 20d0b42885b4d3ebeffbca9ead97b1eb0ddc2fad..88e6e3687ea6f4348ebd5956e677ef26319e33c2 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 0b16c0239b2b6b57087b02f35fd5a5bd9ef857c4..b657f5a463b339a4fbfc8309f79157ed78320f9e 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 6bb30ddbcef8b4bf11b2921feba5bb074069481a..e3e225b9d0fbdb1076a03d4bd12f9fd018f25b90 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 e806f34a08a1b29b70cf95ff8887f349703da0f4..fded52ec97df7d72acaf531eb5f0033c25227669 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 1823531e183f3dbb4e8daaae91c6d9efd22b263e..da29633a794cdb1262d2c307b8c80a94c794830e 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 da5f58d653ea5efb6c00648db56c1bc81a24d44b..ae1c6ab6e0bbd63ebd1369e1741672a6489c87fa 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 1704b651a02d06ff46c8edfd63574d2dce695473..0abe02526ef28041e7c0722fc5aa0b0bc5e7f55d 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 dceccfe85851a9dffe24032c0df5cc199f056ab0..a83f5b95e8988a3d0024ca6d2f3b9601426147cc 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 87e3f831542aac9478566e916d12118f66a7c120..7e9d28d02381dfe36e2a0c7792745618b595588e 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 1ed99d911700c64349816e937230f2cd0e90e437..3012938955f5c9e4fe1e285a0d95478ca7f7b8dd 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 9e78490dc9300fa8d036492449d55e7816daae10..8c07211ed58d7fa2550893fc241a9a2170527a2f 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