Browse Source

Merge branch 'yigit'

pull/2/head
Yiğit Çolakoğlu 6 years ago
parent
commit
a87cd928e4
24 changed files with 475 additions and 110 deletions
  1. +2
    -0
      MyCity/app/build.gradle
  2. +1
    -1
      MyCity/app/src/main/AndroidManifest.xml
  3. +1
    -1
      MyCity/app/src/main/java/gq/yigit/mycity/FileActions.java
  4. +45
    -0
      MyCity/app/src/main/java/gq/yigit/mycity/ImageDownload.java
  5. +19
    -18
      MyCity/app/src/main/java/gq/yigit/mycity/MainActivity.java
  6. +0
    -4
      MyCity/app/src/main/java/gq/yigit/mycity/VotesActivity.java
  7. +2
    -1
      MyCity/app/src/main/java/gq/yigit/mycity/WebRequest.java
  8. +7
    -0
      MyCity/app/src/main/java/gq/yigit/mycity/imageListener.java
  9. +77
    -0
      MyCity/app/src/main/java/gq/yigit/mycity/vote/MyVotesRecyclerViewAdapter.java
  10. +40
    -0
      MyCity/app/src/main/java/gq/yigit/mycity/vote/VotesContent.java
  11. +137
    -0
      MyCity/app/src/main/java/gq/yigit/mycity/vote/VotesFragment.java
  12. +0
    -0
      MyCity/app/src/main/res/drawable-v21/bus.xml
  13. +0
    -0
      MyCity/app/src/main/res/drawable-v21/navigation.xml
  14. +0
    -0
      MyCity/app/src/main/res/drawable-v21/parking.xml
  15. +0
    -0
      MyCity/app/src/main/res/drawable-v21/star.xml
  16. +0
    -0
      MyCity/app/src/main/res/drawable-v21/utility.xml
  17. +0
    -0
      MyCity/app/src/main/res/drawable-v21/vote.xml
  18. +12
    -0
      MyCity/app/src/main/res/drawable/vote_style.xml
  19. +1
    -9
      MyCity/app/src/main/res/layout/app_bar_main.xml
  20. +37
    -0
      MyCity/app/src/main/res/layout/fragment_votes.xml
  21. +18
    -0
      MyCity/app/src/main/res/layout/fragment_votes_list.xml
  22. +0
    -14
      MyCity/app/src/main/res/layout/voting_content.xml
  23. +1
    -0
      MyCity/app/src/main/res/values/dimens.xml
  24. +75
    -62
      server_side/api/voting_system/app.py

+ 2
- 0
MyCity/app/build.gradle View File

@ -25,6 +25,8 @@ dependencies {
implementation 'com.android.support:appcompat-v7:27.0.0' implementation 'com.android.support:appcompat-v7:27.0.0'
implementation 'com.android.support:design:27.0.0' implementation 'com.android.support:design:27.0.0'
implementation 'com.android.support.constraint:constraint-layout:1.1.3' implementation 'com.android.support.constraint:constraint-layout:1.1.3'
implementation 'com.android.support:support-v4:27.1.1'
implementation 'com.android.support:recyclerview-v7:27.1.1'
testImplementation 'junit:junit:4.12' testImplementation 'junit:junit:4.12'
androidTestImplementation 'com.android.support.test:runner:1.0.2' androidTestImplementation 'com.android.support.test:runner:1.0.2'
androidTestImplementation 'com.android.support.test.espresso:espresso-core:3.0.2' androidTestImplementation 'com.android.support.test.espresso:espresso-core:3.0.2'


+ 1
- 1
MyCity/app/src/main/AndroidManifest.xml View File

@ -1,7 +1,7 @@
<?xml version="1.0" encoding="utf-8"?> <?xml version="1.0" encoding="utf-8"?>
<manifest xmlns:android="http://schemas.android.com/apk/res/android" <manifest xmlns:android="http://schemas.android.com/apk/res/android"
package="gq.yigit.mycity"> package="gq.yigit.mycity">
<uses-permission android:name="android.permission.INTERNET" />
<application <application
android:allowBackup="true" android:allowBackup="true"
android:icon="@mipmap/ic_launcher" android:icon="@mipmap/ic_launcher"


MyCity/app/src/main/java/gq/yigit/mycity/fileActions.java → MyCity/app/src/main/java/gq/yigit/mycity/FileActions.java View File


+ 45
- 0
MyCity/app/src/main/java/gq/yigit/mycity/ImageDownload.java View File

@ -0,0 +1,45 @@
package gq.yigit.mycity;
import android.graphics.Bitmap;
import android.graphics.BitmapFactory;
import android.os.AsyncTask;
import android.util.Log;
import java.io.InputStream;
import java.util.ArrayList;
import java.util.List;
public class ImageDownload extends AsyncTask<String, Void, Bitmap> {
private List<imageListener> listeners = new ArrayList<>();
@Override
protected Bitmap doInBackground(String... URL) {
String imageURL = URL[0];
Log.d("[BOOKMARK]",imageURL);
Bitmap bitmap = null;
try {
// Download Image from URL
InputStream input = new java.net.URL(imageURL).openStream();
// Decode Bitmap
bitmap = BitmapFactory.decodeStream(input);
} catch (Exception e) {
e.printStackTrace();
}
return bitmap;
}
@Override
protected void onPostExecute(Bitmap result) {
for (imageListener hl : listeners)
hl.imageDownloaded(result);
}
public void addListener(imageListener toAdd) {
listeners.add(toAdd);
}
}

+ 19
- 18
MyCity/app/src/main/java/gq/yigit/mycity/MainActivity.java View File

@ -3,9 +3,10 @@ package gq.yigit.mycity;
import android.content.Context; import android.content.Context;
import android.content.DialogInterface; import android.content.DialogInterface;
import android.os.Bundle; import android.os.Bundle;
import android.support.constraint.ConstraintLayout;
import android.support.design.widget.FloatingActionButton; import android.support.design.widget.FloatingActionButton;
import android.support.design.widget.Snackbar; import android.support.design.widget.Snackbar;
import android.support.v4.app.FragmentManager;
import android.support.v4.app.FragmentTransaction;
import android.support.v7.app.AlertDialog; import android.support.v7.app.AlertDialog;
import android.util.Log; import android.util.Log;
import android.view.*; import android.view.*;
@ -16,13 +17,11 @@ import android.support.v7.app.ActionBarDrawerToggle;
import android.support.v7.app.AppCompatActivity; import android.support.v7.app.AppCompatActivity;
import android.support.v7.widget.Toolbar; import android.support.v7.widget.Toolbar;
import android.widget.EditText; import android.widget.EditText;
import android.widget.RelativeLayout;
import android.widget.ViewFlipper;
import java.io.*;
import gq.yigit.mycity.vote.VotesContent;
import gq.yigit.mycity.vote.VotesFragment;
public class MainActivity extends AppCompatActivity public class MainActivity extends AppCompatActivity
implements NavigationView.OnNavigationItemSelectedListener {
implements NavigationView.OnNavigationItemSelectedListener, VotesFragment.OnListFragmentInteractionListener {
public Context cntxt; public Context cntxt;
@ -71,23 +70,21 @@ public class MainActivity extends AppCompatActivity
@Override @Override
public boolean onOptionsItemSelected(MenuItem item) { public boolean onOptionsItemSelected(MenuItem item) {
// Handle action bar item clicks here. The action bar will
// automatically handle clicks on the Home/Up button, so long
// as you specify a parent activity in AndroidManifest.xml.
int id = item.getItemId(); int id = item.getItemId();
//noinspection SimplifiableIfStatement //noinspection SimplifiableIfStatement
if (id == R.id.action_settings) { if (id == R.id.action_settings) {
AlertDialog.Builder alert = new AlertDialog.Builder(this); AlertDialog.Builder alert = new AlertDialog.Builder(this);
alert.setTitle("Set new server"); alert.setTitle("Set new server");
alert.setMessage("Server IP:");
alert.setMessage("Server URL:");
final EditText input = new EditText(this); final EditText input = new EditText(this);
alert.setView(input); alert.setView(input);
alert.setPositiveButton("Ok", new DialogInterface.OnClickListener() { alert.setPositiveButton("Ok", new DialogInterface.OnClickListener() {
public void onClick(DialogInterface dialog, int whichButton) { public void onClick(DialogInterface dialog, int whichButton) {
fileActions file_manager = new fileActions();
file_manager.writeToFile(input.getText().toString(),cntxt,"serverip.config");
FileActions file_manager = new FileActions();
file_manager.writeToFile(input.getText().toString(),cntxt,"server.config");
} }
}); });
@ -109,14 +106,14 @@ public class MainActivity extends AppCompatActivity
public boolean onNavigationItemSelected(MenuItem item) { public boolean onNavigationItemSelected(MenuItem item) {
// Handle navigation view item clicks here. // Handle navigation view item clicks here.
int id = item.getItemId(); int id = item.getItemId();
FragmentManager fragmentManager = getSupportFragmentManager();
FragmentTransaction fragmentTransaction = fragmentManager.beginTransaction();
if (id == R.id.voting) { if (id == R.id.voting) {
ViewFlipper vf = (ViewFlipper)findViewById(R.id.page);
try {
vf.setDisplayedChild(1);
}catch (Exception e){
Log.e("[ERROR]",e.toString());
}
VotesFragment fragment = new VotesFragment();
fragmentTransaction.add(R.id.app_bar_main, fragment);
fragmentTransaction.commit();
} else if (id == R.id.parking) { } else if (id == R.id.parking) {
} else if (id == R.id.transit) { } else if (id == R.id.transit) {
@ -134,4 +131,8 @@ public class MainActivity extends AppCompatActivity
return true; return true;
} }
public void onListFragmentInteraction(VotesContent.VoteItem vote){
Log.i("[INFO]",vote.id);
}
} }

+ 0
- 4
MyCity/app/src/main/java/gq/yigit/mycity/VotesActivity.java View File

@ -1,4 +0,0 @@
package gq.yigit.mycity;
public class VotesActivity {
}

+ 2
- 1
MyCity/app/src/main/java/gq/yigit/mycity/WebRequest.java View File

@ -27,7 +27,8 @@ public class WebRequest extends AsyncTask<Void,Void,String> {
private List<responseListener> listeners = new ArrayList<>(); private List<responseListener> listeners = new ArrayList<>();
WebRequest(String url, boolean request_type, HashMap<String,String> request_content){
public WebRequest(String url, boolean request_type, HashMap<String,String> request_content){
//request_type=true:get else post
this.url = url; this.url = url;
this.request_content = request_content; this.request_content = request_content;


+ 7
- 0
MyCity/app/src/main/java/gq/yigit/mycity/imageListener.java View File

@ -0,0 +1,7 @@
package gq.yigit.mycity;
import android.graphics.Bitmap;
public interface imageListener {
public void imageDownloaded(Bitmap img);
}

+ 77
- 0
MyCity/app/src/main/java/gq/yigit/mycity/vote/MyVotesRecyclerViewAdapter.java View File

@ -0,0 +1,77 @@
package gq.yigit.mycity.vote;
import android.support.v7.widget.RecyclerView;
import android.util.Log;
import android.view.LayoutInflater;
import android.view.View;
import android.view.ViewGroup;
import android.widget.ImageView;
import android.widget.TextView;
import gq.yigit.mycity.R;
import gq.yigit.mycity.vote.VotesFragment.OnListFragmentInteractionListener;
import gq.yigit.mycity.vote.VotesContent.VoteItem;
import java.util.List;
public class MyVotesRecyclerViewAdapter extends RecyclerView.Adapter<MyVotesRecyclerViewAdapter.ViewHolder> {
private final List<VoteItem> mValues;
private final OnListFragmentInteractionListener mListener;
public MyVotesRecyclerViewAdapter(List<VoteItem> items, OnListFragmentInteractionListener listener) {
mValues = items;
mListener = listener;
}
@Override
public ViewHolder onCreateViewHolder(ViewGroup parent, int viewType) {
View view = LayoutInflater.from(parent.getContext())
.inflate(R.layout.fragment_votes, parent, false);
return new ViewHolder(view);
}
@Override
public void onBindViewHolder(final ViewHolder holder, int position) {
holder.mItem = mValues.get(position);
holder.mIdView.setText(mValues.get(position).name);
holder.mContentView.setText(mValues.get(position).details);
holder.mImageView.setImageBitmap(mValues.get(position).img);
holder.mView.setOnClickListener(new View.OnClickListener() {
@Override
public void onClick(View v) {
if (null != mListener) {
mListener.onListFragmentInteraction(holder.mItem);
}
}
});
}
@Override
public int getItemCount() {
return mValues.size();
}
public class ViewHolder extends RecyclerView.ViewHolder {
public final View mView;
public final TextView mIdView;
public final TextView mContentView;
public final ImageView mImageView;
public VoteItem mItem;
public ViewHolder(View view) {
super(view);
mView = view;
mIdView = (TextView) view.findViewById(R.id.item_number);
mContentView = (TextView) view.findViewById(R.id.content);
mImageView = (ImageView) view.findViewById(R.id.vote_img);
}
@Override
public String toString() {
return super.toString() + " '" + mContentView.getText() + "'";
}
}
}

+ 40
- 0
MyCity/app/src/main/java/gq/yigit/mycity/vote/VotesContent.java View File

@ -0,0 +1,40 @@
package gq.yigit.mycity.vote;
import android.graphics.Bitmap;
import java.util.ArrayList;
import java.util.HashMap;
import java.util.List;
import java.util.Map;
public class VotesContent {
public static final List<VoteItem> ITEMS = new ArrayList<VoteItem>();
public static final Map<String, VoteItem> ITEM_MAP = new HashMap<String, VoteItem>();
public static void addItem(VoteItem item) {
ITEMS.add(item);
ITEM_MAP.put(item.id, item);
}
public static class VoteItem {
public final String id;
public final String name;
public final String details;
public final Bitmap img;
public VoteItem(String id, String name, String details, Bitmap img) {
this.id = id;
this.name = name;
this.details = details;
this.img = img;
}
@Override
public String toString() {
return name;
}
}
}

+ 137
- 0
MyCity/app/src/main/java/gq/yigit/mycity/vote/VotesFragment.java View File

@ -0,0 +1,137 @@
package gq.yigit.mycity.vote;
import android.content.Context;
import android.graphics.Bitmap;
import android.os.Bundle;
import android.support.v4.app.Fragment;
import android.support.v7.widget.GridLayoutManager;
import android.support.v7.widget.LinearLayoutManager;
import android.support.v7.widget.RecyclerView;
import android.util.Log;
import android.view.LayoutInflater;
import android.view.View;
import android.view.ViewGroup;
import gq.yigit.mycity.*;
import gq.yigit.mycity.vote.VotesContent.VoteItem;
import org.json.JSONArray;
import org.json.JSONObject;
import java.util.HashMap;
public class VotesFragment extends Fragment implements responseListener, imageListener {
private static final String ARG_COLUMN_COUNT = "column-count";
private int mColumnCount = 1;
private OnListFragmentInteractionListener mListener;
public RecyclerView recyclerView;
public String url;
public int img_count = 0;
public JSONArray votes;
public ImageDownload img_downloader;
public int j = 0;
public VotesFragment() {
}
public static VotesFragment newInstance(int columnCount) {
VotesFragment fragment = new VotesFragment();
Bundle args = new Bundle();
args.putInt(ARG_COLUMN_COUNT, columnCount);
fragment.setArguments(args);
return fragment;
}
@Override
public void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
if (getArguments() != null) {
mColumnCount = getArguments().getInt(ARG_COLUMN_COUNT);
}
}
@Override
public View onCreateView(LayoutInflater inflater, ViewGroup container,
Bundle savedInstanceState) {
View view = inflater.inflate(R.layout.fragment_votes_list, container, false);
// Set the adapter
if (view instanceof RecyclerView) {
Context context = view.getContext();
recyclerView = (RecyclerView) view;
if (mColumnCount <= 1) {
recyclerView.setLayoutManager(new LinearLayoutManager(context));
} else {
recyclerView.setLayoutManager(new GridLayoutManager(context, mColumnCount));
}
FileActions file_manager = new FileActions();
url = (file_manager.readFromFile(context,"server.config")).trim();
WebRequest web_manager = new WebRequest(url + "/votings",true,new HashMap<String,String>());
web_manager.addListener(this);
web_manager.execute();
}
return view;
}
@Override
public void onAttach(Context context) {
super.onAttach(context);
if (context instanceof OnListFragmentInteractionListener) {
mListener = (OnListFragmentInteractionListener) context;
} else {
throw new RuntimeException(context.toString()
+ " must implement OnListFragmentInteractionListener");
}
}
@Override
public void onDetach() {
super.onDetach();
mListener = null;
}
public interface OnListFragmentInteractionListener {
void onListFragmentInteraction(VoteItem item);
}
public void receivedResponse(boolean success, String response){
if(success) {
try {
votes = new JSONArray(response);
img_downloader = new ImageDownload();
img_downloader.addListener(this);
img_downloader.execute(url+ ((JSONObject)votes.get(j)).getString("img"));
}catch (Exception e){
Log.e("[ERROR]:",e.getMessage());
}
Log.i("[INFO]",VotesContent.ITEMS.toString());
}else{
Log.e("[ERROR]:",response);
}
}
public void imageDownloaded(Bitmap img){
try {
JSONObject vote = (JSONObject) votes.get(j);
vote.put("img",img);
j++;
if(j>votes.length()-1) {
for (int i = 0; i < votes.length(); i++) {
JSONObject poll = (JSONObject) votes.get(i);
VotesContent.addItem(new VoteItem(poll.get("id").toString(), poll.get("name").toString(), poll.get("desc").toString(), (Bitmap) poll.get("img")));
}
recyclerView.setAdapter(new MyVotesRecyclerViewAdapter(VotesContent.ITEMS, mListener));
}else{
img_downloader = new ImageDownload();
img_downloader.addListener(this);
img_downloader.execute(url+ ((JSONObject)votes.get(j)).getString("img"));
}
} catch (Exception e) {
Log.e("[ERROR]", e.toString());
}
}
}

MyCity/app/src/main/res/drawable/bus.xml → MyCity/app/src/main/res/drawable-v21/bus.xml View File


MyCity/app/src/main/res/drawable/navigation.xml → MyCity/app/src/main/res/drawable-v21/navigation.xml View File


MyCity/app/src/main/res/drawable/parking.xml → MyCity/app/src/main/res/drawable-v21/parking.xml View File


MyCity/app/src/main/res/drawable/star.xml → MyCity/app/src/main/res/drawable-v21/star.xml View File


MyCity/app/src/main/res/drawable/utility.xml → MyCity/app/src/main/res/drawable-v21/utility.xml View File


MyCity/app/src/main/res/drawable/vote.xml → MyCity/app/src/main/res/drawable-v21/vote.xml View File


+ 12
- 0
MyCity/app/src/main/res/drawable/vote_style.xml View File

@ -0,0 +1,12 @@
<?xml version="1.0" encoding="utf-8"?>
<shape xmlns:android="http://schemas.android.com/apk/res/android"
android:shape="rectangle">
<corners
android:radius="2dp"
android:topRightRadius="0dp"
android:bottomRightRadius="0dp"
android:bottomLeftRadius="0dp"/>
<stroke
android:width="1dp"
android:color="@android:color/black" />
</shape>

+ 1
- 9
MyCity/app/src/main/res/layout/app_bar_main.xml View File

@ -5,6 +5,7 @@
xmlns:tools="http://schemas.android.com/tools" xmlns:tools="http://schemas.android.com/tools"
android:layout_width="match_parent" android:layout_width="match_parent"
android:layout_height="match_parent" android:layout_height="match_parent"
android:id="@+id/app_bar_main"
tools:context=".MainActivity"> tools:context=".MainActivity">
<android.support.design.widget.AppBarLayout <android.support.design.widget.AppBarLayout
@ -20,15 +21,6 @@
app:popupTheme="@style/AppTheme.PopupOverlay"/> app:popupTheme="@style/AppTheme.PopupOverlay"/>
</android.support.design.widget.AppBarLayout> </android.support.design.widget.AppBarLayout>
<ViewFlipper xmlns:android="http://schemas.android.com/apk/res/android"
android:id="@+id/page"
android:layout_width="fill_parent"
android:layout_height="wrap_content" >
<include android:id="@+id/include1" layout="@layout/voting_content" />
</ViewFlipper>
<android.support.design.widget.FloatingActionButton <android.support.design.widget.FloatingActionButton
android:id="@+id/fab" android:id="@+id/fab"
android:layout_width="wrap_content" android:layout_width="wrap_content"


+ 37
- 0
MyCity/app/src/main/res/layout/fragment_votes.xml View File

@ -0,0 +1,37 @@
<?xml version="1.0" encoding="utf-8"?>
<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
xmlns:app="http://schemas.android.com/apk/res-auto" android:layout_width="match_parent"
android:layout_height="wrap_content"
android:orientation="horizontal"
android:background="@drawable/vote_style"
android:layout_marginBottom="10dp">
<ImageView
android:layout_width="150dp"
android:layout_height="50dp" app:srcCompat="@mipmap/ic_launcher" android:id="@+id/vote_img"
android:layout_weight="1"/>
<LinearLayout
android:orientation="vertical"
android:layout_width="match_parent"
android:layout_height="wrap_content" android:layout_weight="1">
<TextView
android:id="@+id/item_number"
android:layout_width="match_parent"
android:layout_height="wrap_content"
android:layout_margin="4dp"
android:textAppearance="?attr/textAppearanceListItem" android:textSize="14sp"/>
<LinearLayout
android:orientation="vertical"
android:layout_width="match_parent"
android:layout_height="wrap_content" android:layout_weight="1"
android:layout_marginLeft="16dp">
<TextView
android:id="@+id/content"
android:layout_width="match_parent"
android:layout_height="wrap_content"
android:layout_margin="4dp"
android:textAppearance="?attr/textAppearanceListItem" android:textColor="#b5b5b5"
android:textSize="12sp"/>
</LinearLayout>
</LinearLayout>
</LinearLayout>

+ 18
- 0
MyCity/app/src/main/res/layout/fragment_votes_list.xml View File

@ -0,0 +1,18 @@
<?xml version="1.0" encoding="utf-8"?>
<android.support.v7.widget.RecyclerView
xmlns:android="http://schemas.android.com/apk/res/android"
xmlns:app="http://schemas.android.com/apk/res-auto"
xmlns:tools="http://schemas.android.com/tools"
android:id="@+id/list"
android:name="gq.yigit.mycity.VotesFragment"
android:layout_width="match_parent"
android:layout_height="wrap_content"
android:layout_marginLeft="16dp"
android:layout_marginTop="60dp"
android:layout_marginRight="16dp"
app:layoutManager="LinearLayoutManager"
tools:context=".vote.VotesFragment"
tools:listitem="@layout/fragment_votes"
android:divider="@android:color/transparent"
android:dividerHeight="100.0sp"
android:clipChildren="false" android:clipToPadding="false" android:scrollbars="vertical"/>

+ 0
- 14
MyCity/app/src/main/res/layout/voting_content.xml View File

@ -1,14 +0,0 @@
<?xml version="1.0" encoding="utf-8"?>
<android.support.constraint.ConstraintLayout xmlns:android="http://schemas.android.com/apk/res/android"
xmlns:app="http://schemas.android.com/apk/res-auto" xmlns:tools="http://schemas.android.com/tools" android:layout_width="match_parent"
android:layout_height="match_parent">
<TextView
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:text="Bonjour!"
app:layout_constraintBottom_toBottomOf="parent"
app:layout_constraintLeft_toLeftOf="parent"
app:layout_constraintRight_toRightOf="parent"
app:layout_constraintTop_toTopOf="parent"/>
</android.support.constraint.ConstraintLayout>

+ 1
- 0
MyCity/app/src/main/res/values/dimens.xml View File

@ -5,4 +5,5 @@
<dimen name="nav_header_vertical_spacing">8dp</dimen> <dimen name="nav_header_vertical_spacing">8dp</dimen>
<dimen name="nav_header_height">176dp</dimen> <dimen name="nav_header_height">176dp</dimen>
<dimen name="fab_margin">16dp</dimen> <dimen name="fab_margin">16dp</dimen>
<dimen name="text_margin">16dp</dimen>
</resources> </resources>

+ 75
- 62
server_side/api/voting_system/app.py View File

@ -1,84 +1,97 @@
import os import os
import ssl
import json import json
from flask import Flask, send_from_directory
from flask_restful import Resource, Api, abort, reqparse
from flask import Flask, send_from_directory, request
from flask_restful import Resource, Api, abort
app = Flask(__name__) app = Flask(__name__)
api = Api(app) api = Api(app)
parser = reqparse.RequestParser()
parser.add_argument('name', required=True)
parser.add_argument('desc')
parser.add_argument('img')
parser.add_argument('votes', required=True)
with open(os.path.join(app.root_path, 'votings.json'), 'r') as f: with open(os.path.join(app.root_path, 'votings.json'), 'r') as f:
votings = json.load(f)
votings = json.load(f)
class Votings(Resource): class Votings(Resource):
def get(self):
voting = [
{
'id' : v['id'],
'name': v['name'],
'desc': v['desc'],
'img' : v['img']
}
for v in votings
]
return voting
def post(self):
args = parser.parse_args()
voting_id = len(votings) + 1
voting = {
'id': voting_id,
'name': args['name'],
'desc': args['desc'],
'img' : args['img'],
'votes': [
{
'id': k,
'name': vote['name'],
'desc': vote['desc'],
'votes': 0
}
for k, vote in enumerate(json.loads(args['votes']))
]
}
votings.append(voting)
with open(os.path.join(app.root_path, 'votings.json'), 'w') as f:
json.dump(votings, f, indent=4)
return voting
def get(self):
voting = [
{
'id' : v['id'],
'name': v['name'],
'desc': v['desc'],
'img' : v['img']
}
for v in votings
]
return voting
def post(self):
"""
Example POST Data:
name=<voting_name>&
desc=<voting_desc>& # OPTIONAL
img=<voting_img>& # OPTIONAL
votes=[
{
"name": "<vote_name>",
"desc": "<vote_desc>" # OPTIONAL
},
(...)
]
"""
args = request.form
voting_id = len(votings) + 1
voting = {
'id': voting_id,
'name': args['name'],
'desc': args.get('desc'),
'img' : args.get('img'),
'votes': [
{
'id' : k,
'name': vote['name'],
'desc': vote.get('desc'),
'votes': 0
}
for k, vote in enumerate(json.loads(args['votes']))
]
}
votings.append(voting)
with open(os.path.join(app.root_path, 'votings.json'), 'w') as f:
json.dump(votings, f, indent=4)
return voting
class Voting(Resource): class Voting(Resource):
def get(self, voting_id):
try:
return votings[voting_id - 1]
except:
abort(404, error="Voting {} doesn't exist".format(voting_id))
def get(self, voting_id):
try:
return votings[voting_id - 1]
except:
abort(404, error="Voting {} doesn't exist".format(voting_id))
class Vote(Resource): class Vote(Resource):
def get(self, voting_id, vote_id):
votings[voting_id - 1]['votes'][vote_id - 1]['votes'] += 1
with open(os.path.join(app.root_path, 'votings.json'), 'w') as f:
json.dump(votings, f, indent=4)
return votings[voting_id - 1]
def get(self):
"""
Example URL Query:
/vote?voting_id=<voting_id>&vote_id=<vote_id>
"""
voting_id = int(request.args['voting_id'])
vote_id = int(request.args['vote_id'])
votings[voting_id - 1]['votes'][vote_id - 1]['votes'] += 1
with open(os.path.join(app.root_path, 'votings.json'), 'w') as f:
json.dump(votings, f, indent=4)
return votings[voting_id - 1]
api.add_resource(Votings, '/votings', '/votings/') api.add_resource(Votings, '/votings', '/votings/')
api.add_resource(Voting, '/votings/<int:voting_id>') api.add_resource(Voting, '/votings/<int:voting_id>')
api.add_resource(Vote, '/vote/<int:voting_id>/<int:vote_id>')
api.add_resource(Vote, '/vote', '/vote/')
@app.route('/img/<path:path>') @app.route('/img/<path:path>')
def send_img(path): def send_img(path):
return send_from_directory('images', path)
return send_from_directory('images', path)
if __name__ == '__main__': if __name__ == '__main__':
app.run(host='0.0.0.0', port=5000, debug=True)
app.run(host='0.0.0.0', port=5000)

Loading…
Cancel
Save