- App Name: Our app's name is "score!".
- App Tagline: All things Cornell Sports!
- The ios team members' repository can be found here: https://github.com/erin-xu/sports-tracker
- The screenshots below demonstrate the functionality and show the user interface of the app.
When the app is launched, the home screen is shown. On this screen, the next sports event is displayed at the top, along with the team, opponent, date, time, and location. More upcoming events are displayed below in a horizontally-scrollable collection view, including the date, team, opponent, and location. The “see all” button can be clicked to open the upcoming events page.
On this page, all upcoming events can be viewed and searched through using gender and sport filters. The team, location, time, date, and opponent is displayed for each upcoming event.
Sports events can also be browsed by team. When clicking on a sports team on the home page, a new page opens with a complete schedule of past and future games. The date, opponent, and location is displayed for each of these events. If the event has already occurred, the result is displayed as well.
-
App Description: An app designed for all Cornell Sports enthusiasts, score! helps its users keep track of Cornell Sporting events. The purpose of score! to make tracking and keeping up with Cornell sports easier for Cornell's incredibly busy students. It is an app which, on its backend, maintains an API of Cornell sporting events, Cornell teams, and users. A team can host many events, and each event is owned by exactly one Cornell team. The backend provides an API which holds all the events that teams have posted, serialized into accessible and readable data about the event for the frontend. On the frontend, data is retrieved from the provided API, and organized so users can filter, sort, and view both completed and upcoming Cornell sporting events. Users are able to filter upcoming events by both type of sport and the team's gender (i.e. Men's Soccer vs. Women's Soccer). Users are also able to access a list of a specific team's upcoming and completed events. When viewing an event, users see the time, date, and location of the event (and if the event has been completed, the score and whether Cornell won or not).
-
iOS Requirements: Utilizes NSLayoutConstraint. Implements UICollectionView. Uses UINavigationController to navigate between screens. Integrates the API provided by the backend for data retrieval regarding the information for the sport evnets.
Backend Requirements: The backend provided an API which both met and passed the outlined requirements. Our API implements 14 routes (including GET, POST, and DELETE methods), where 4 are required (note: the frontend does not yet utilize all provided routes, but does use more than 4 of the required routes). Our database houses three tables (Fan, Team, and Event Tables) and an association table between the Fan and Team tables, as well as relational keys between the Event and Team tables. Finally, an API specification is provided below for all 14 of the routes (both those utilized and not yet utilized by the frontend).
-
Other Notes: The backend implemented routes that haven't yet been implemented by the frontend. We believe in the future these routes could be used to enhance the features and functionality of the app. For example, the capcity for user's to create password-protected accounts and to select favorite teams already exists within the backend.
"/api/teams/", methods=["GET"]
Response:
<HTTP STATUS CODE 200>
{
"teams": [
{
"id": 1,
"name": "Cornell Women's Soccer",
"gender": "Women's",
"sport": "Soccer",
"events": [<SERIALIZED EVENT WITHOUT TEAM FIELD>, ... ]
},
{
"id": 2,
"name": "Cornell Men's Soccer",
"gender": "Men's",
"sport": "Soccer",
"events": [<SERIALIZED EVENT WITHOUT TEAM FIELD>, ... ]
}
...
]
}
"/api/teams/", methods=["POST"]
Request:
{
"name": "Cornell Women's Soccer",
"password": "password1",
"gender": "Women's",
"sport": "Soccer",
}
Response:
<HTTP STATUS CODE 201>
{
"id": 1,
"name": "Cornell Women's Soccer",
"gender": "Women's",
"sport": "Soccer",
"events": [ <SERIALIZED EVENT WITHOUT TEAM FIELD>, ... ]
}
"/api/teams/<int:team_id>/", methods=["GET"]
Response:
<HTTP STATUS CODE 200>
{
"id": 1,
"name": "Cornell Women's Soccer",
"gender": "Women's",
"sport": "Soccer",
"events": [ <SERIALIZED EVENT WITHOUT TEAM FIELD>, ... ]
}
"/api/teams/<int:team_id>/", methods=["DELETE"]
Request:
{
"name": "Cornell Women's Soccer",
"password": "password1"
}
Response:
<HTTP STATUS CODE 200>
{
"id": 1,
"name": "Cornell Women's Soccer",
"gender": "Women's",
"sport": "Soccer",
"events": [ <SERIALIZED EVENT WITHOUT TEAM FIELD>, ... ]
}
"/api/events/", methods=["GET"]
Response:
<HTTP STATUS CODE 200>
{
"events": [
{
"id": 1,
"title": "Cornell Women's Soccer vs. UVA",
"unixTime": 1638485544, // Event beginning in unix time
"location": "Charles F. Berman Field",
"date": "2021-08-02",
"time": "06:43:43",
"description": "Enjoy pizza and root for our undefeated soccer team.",
"opponent": "UVA Women's Soccer",
"score": "3-1",
"win": "W",
"team": <SERIALIZED TEAM WITHOUT EVENT FIELD>
},
{
"id": 2,
"title": "Princeton Hockey vs. Cornell Hockey",
"unixTime": 1553364809, // Event beginning in unix time
"location": "Lynah Rink",
"date": "2021-08-02"
"time": "06:43:43",
"description": "Don't miss the game of the century!",
"opponent": "Princeton Hockey"
"score": "1-2",
"win": "L",
"team": <SERIALIZED TEAM WITHOUT EVENT FIELD>
}
...
]
}
"/api/events/<int:event_id>", methods=["GET"]
Response:
<HTTP STATUS CODE 200>
{
"id": 1,
"title": "Cornell Women's Soccer vs. UVA",
"unixTime": 1638485544, // Event beginning in unix time
"location": "Charles F. Berman Field",
"date": "2021-08-02",
"time": "06:43:43",
"description": "Enjoy pizza and root for our undefeated soccer team.",
"opponent": "UVA Women's Soccer",
"score": "3-1",
"win": "W",
"team": <SERIALIZED TEAM WITHOUT EVENT FIELD>
}
"/api/events/", methods=["POST"]
Request:
{
"team_id": 3,
"name": "Cornell Women's Soccer",
"password": "password1"
"title": "Cornell Women's Soccer vs. UVA",
"unixTime": 1638485544, // Event beginning in unix time
"location": "Charles F. Berman Field",
"description": "Enjoy pizza and root for our undefeated soccer team.",
"opponent": "UVA Women's Soccer",
"score": "3-1",
"win": 0
}
Response:
<HTTP STATUS CODE 201>
{
"id": 1,
"title": "Cornell Women's Soccer vs. UVA",
"unixTime": 1638485544, // Event beginning in unix time
"location": "Charles F. Berman Field",
"date": "2021-08-02",
"time": "06:43:43",
"description": "Enjoy pizza and root for our undefeated soccer team.",
"opponent": "UVA Women's Soccer",
"score": "3-1",
"win": "W",
"team": <SERIALIZED TEAM WITHOUT EVENT FIELD>
}
"/api/events/", methods=["POST"]
Request:
{
"name": "Cornell Women's Soccer",
"password": "password1"
"title": "Cornell Women's Soccer vs. UVA",
"unixTime": 1638485544, // Event beginning in unix time
"location": "Charles F. Berman Field",
"description": "Enjoy pizza and root for our undefeated soccer team.",
"opponent": "UVA Women's Soccer",
"score": "3-1",
"win": 0
}
Response:
<HTTP STATUS CODE 201>
{
"id": 1,
"title": "Cornell Women's Soccer vs. UVA",
"unixTime": 1638485544, // Event beginning in unix time
"location": "Charles F. Berman Field",
"date": "2021-08-02",
"time": "06:43:43",
"description": "Enjoy pizza and root for our undefeated soccer team.",
"opponent": "UVA Women's Soccer",
"score": "3-1",
"win": "W",
"team": <SERIALIZED TEAM WITHOUT EVENT FIELD>
}
"/api/events/<int:event_id>", methods=["DELETE"]
Request:
{
"name": "Cornell Women's Soccer",
"password": "password1"
}
Response:
<HTTP STATUS CODE 201>
{
"id": 1,
"title": "Cornell Women's Soccer vs. UVA",
"unixTime": 1638485544, // Event beginning in unix time
"location": "Charles F. Berman Field",
"date": "2021-08-02",
"time": "06:43:43",
"description": "Enjoy pizza and root for our undefeated soccer team.",
"opponent": "UVA Women's Soccer",
"score": "3-1",
"win": "W",
"team": <SERIALIZED TEAM WITHOUT EVENT FIELD>
}
"/api/events/users/<int:user_id>/", methods=["GET"]
Request:
{
"name": "John Smith",
"password": "iluvmymommy"
}
Response:
<HTTP STATUS CODE 200>
{
"events": [
{
"id": 3,
"title": "Cornell Women's Soccer vs. UVA",
"unixTime": 1638485544, // Event beginning in unix time
"location": "Charles F. Berman Field",
"date": "2021-08-02",
"time": "06:43:43",
"description": "Enjoy pizza and root for our undefeated soccer team.",
"opponent": "UVA Women's Soccer",
"score": "3-1",
"win": "W",
"team": <SERIALIZED TEAM WITHOUT EVENT FIELD>
},
{
"id": 7,
"title": "Princeton Hockey vs. Cornell Hockey",
"unixTime": 1553364809, // Event beginning in unix time
"location": "Lynah Rink",
"date": "2021-08-02"
"time": "06:43:43",
"description": "Don't miss the game of the century!",
"opponent": "Princeton Hockey"
"score": "1-2",
"win": "L",
"team": <SERIALIZED TEAM WITHOUT EVENT FIELD>
}
...
]
}
"/api/users/<int:user_id>/", methods=["GET"]
Request:
{
"name": "John Smith",
"password": "iluvmymommy"
}
Response:
<HTTP STATUS CODE 200>
{
"id": 41,
"name": "John Smith",
"teams": [<SERIALIZED TEAM>, ... ] // Favorite teams
}
"/api/users/", methods=["POST"]
Request:
{
"name": "Inle Bush",
"password": "supersecretpassword"
}
Response:
<HTTP STATUS CODE 201>
{
"id": 30,
"name": "Inle Bush",
"teams": []
}
"/api/users/<int:user_id>/favorites/<int:team_id>/", methods=["POST"]
Request:
{
"name": "John Smith",
"password": "iluvmymommy"
}
Response:
<HTTP STATUS CODE 201>
{
"id": 41,
"name": "John Smith",
"teams": [<SERIALIZED TEAM>, ... ] // Favorite teams with added team
}
"/api/users/<int:user_id>/remove/<int:team_id>/", methods=["POST"]
Request:
{
"name": "John Smith",
"password": "iluvmymommy"
}
Response:
<HTTP STATUS CODE 200>
{
"id": 41,
"name": "John Smith",
"teams": [<SERIALIZED TEAM>, ... ] // Favorite teams without team
}