From 23ab01b374bb71ba02a183195de3c6cd343b5d5a Mon Sep 17 00:00:00 2001 From: Michael Hall Date: Mon, 16 Apr 2018 21:52:24 -0400 Subject: [PATCH] Rework the new team creation workflow to make it simpler and include a little more instruction. If a user doesn't have a team near them, prompt them to create one instead of showing them an empty homepage. Also impoves some of the instructions around creating a new event. Fixes #55 --- events/forms.py | 7 +- events/location.py | 22 ++++-- events/models/profiles.py | 2 +- events/templates/events/define_team_form.html | 3 + events/templates/events/new_team_form.html | 15 ++++ .../get_together/events/create_event.html | 2 +- .../get_together/new_team/define_team.html | 19 +++++ .../get_together/new_team/start_new_team.html | 49 +++++++++++++ .../get_together/places/create_place.html | 2 +- .../templates/get_together/users/login.html | 13 ++-- get_together/urls.py | 3 +- get_together/views/__init__.py | 10 ++- get_together/views/events.py | 1 + get_together/views/new_team.py | 71 +++++++++++++++++++ 14 files changed, 201 insertions(+), 18 deletions(-) create mode 100644 events/templates/events/define_team_form.html create mode 100644 events/templates/events/new_team_form.html create mode 100644 get_together/templates/get_together/new_team/define_team.html create mode 100644 get_together/templates/get_together/new_team/start_new_team.html create mode 100644 get_together/views/new_team.py diff --git a/events/forms.py b/events/forms.py index b278144..ad28d25 100644 --- a/events/forms.py +++ b/events/forms.py @@ -153,7 +153,7 @@ class TeamForm(forms.ModelForm): class NewTeamForm(forms.ModelForm): class Meta: model = Team - fields = ['name', 'description', 'category', 'city', 'web_url', 'tz'] + fields = ['name', 'city', 'tz'] widgets = { 'city': Lookup(source=City), } @@ -162,6 +162,11 @@ class NewTeamForm(forms.ModelForm): super().__init__(*args, **kwargs) self.fields['city'].required = True +class TeamDefinitionForm(forms.ModelForm): + class Meta: + model = Team + fields = ['category', 'web_url', 'description'] + class DeleteTeamForm(forms.Form): confirm = forms.BooleanField(label="Yes, delete team", required=True) diff --git a/events/location.py b/events/location.py index ebe25ca..ab84ea8 100644 --- a/events/location.py +++ b/events/location.py @@ -1,6 +1,8 @@ from django.utils import timezone from django.conf import settings +from .models.locale import City + import math import pytz import datetime @@ -27,8 +29,8 @@ def get_geoip(request): client_ip = get_client_ip(request) if client_ip == '127.0.0.1' or client_ip == 'localhost': if settings.DEBUG: - client_ip = '8.8.8.8' # Try Google's server - print("Client is localhost, using 8.8.8.8 for geoip instead") + client_ip = getattr(settings, 'DEBUG_IP', '8.8.8.8') # Try Google's server + print("Client is localhost, using %s for geoip instead" % client_ip) else: raise Exception("Client is localhost") @@ -47,7 +49,6 @@ def distance(center1, center2): dlat = (center2[0] - center1[0]) * KM_PER_DEGREE_LAT dlng = (center2[1] - center1[1]) * (KM_PER_DEGREE_LNG*math.cos(math.radians(avglat))) dkm = math.sqrt((dlat*dlat) + (dlng*dlng)) - print("Distance between %s and %s is %s" % (center1, center2, dkm)) return dkm def city_distance_from(ll, city): @@ -76,4 +77,17 @@ def searchable_distance_from(ll, searchable): else: return 99999 - +def get_nearest_city(ll, max_distance=100): + city = None + city_distance = 1 #km + while city is None and city_distance <= max_distance: + minlat = ll[0]-(city_distance/KM_PER_DEGREE_LAT) + maxlat = ll[0]+(city_distance/KM_PER_DEGREE_LAT) + minlng = ll[1]-(city_distance/(KM_PER_DEGREE_LNG*math.cos(math.radians(ll[0])))) + maxlng = ll[1]+(city_distance/(KM_PER_DEGREE_LNG*math.cos(math.radians(ll[0])))) + nearby_cities = City.objects.filter(latitude__gte=minlat, latitude__lte=maxlat, longitude__gte=minlng, longitude__lte=maxlng) + if len(nearby_cities) == 0: + city_distance += 1 + else: + return sorted(nearby_cities, key=lambda city: city_distance_from(ll, city))[0] + return city diff --git a/events/models/profiles.py b/events/models/profiles.py index 06b12cc..ab9bfbf 100644 --- a/events/models/profiles.py +++ b/events/models/profiles.py @@ -197,7 +197,7 @@ class Team(models.Model): name = models.CharField(_("Team Name"), max_length=256, null=False, blank=False) organization = models.ForeignKey(Organization, related_name='teams', null=True, blank=True, on_delete=models.CASCADE) - description = models.TextField(help_text=_('Team Description'), blank=True, null=True) + description = models.TextField(blank=True, null=True) country = models.ForeignKey(Country, on_delete=models.CASCADE) spr = models.ForeignKey(SPR, null=True, blank=True, on_delete=models.CASCADE) diff --git a/events/templates/events/define_team_form.html b/events/templates/events/define_team_form.html new file mode 100644 index 0000000..3ff8ff6 --- /dev/null +++ b/events/templates/events/define_team_form.html @@ -0,0 +1,3 @@ + + {{ team_form }} +
diff --git a/events/templates/events/new_team_form.html b/events/templates/events/new_team_form.html new file mode 100644 index 0000000..2bab003 --- /dev/null +++ b/events/templates/events/new_team_form.html @@ -0,0 +1,15 @@ + + + + + + + + + + + + + + +
{{ team_form.name }}

{{ team_form.city }}
{{ team_form.tz }}
diff --git a/get_together/templates/get_together/events/create_event.html b/get_together/templates/get_together/events/create_event.html index ac7043d..f9d8fc8 100644 --- a/get_together/templates/get_together/events/create_event.html +++ b/get_together/templates/get_together/events/create_event.html @@ -21,7 +21,7 @@
{% include "events/event_form.html" %}
- +
{% endblock %} diff --git a/get_together/templates/get_together/new_team/define_team.html b/get_together/templates/get_together/new_team/define_team.html new file mode 100644 index 0000000..e38e42e --- /dev/null +++ b/get_together/templates/get_together/new_team/define_team.html @@ -0,0 +1,19 @@ +{% extends "get_together/base.html" %} +{% load static %} + +{% block content %} +
+
+
+
+

Tell people more about your team

+
+ {% csrf_token %} +

{% include "events/define_team_form.html" %}

+

+
+
+
+
+
+{% endblock %} diff --git a/get_together/templates/get_together/new_team/start_new_team.html b/get_together/templates/get_together/new_team/start_new_team.html new file mode 100644 index 0000000..6a0313d --- /dev/null +++ b/get_together/templates/get_together/new_team/start_new_team.html @@ -0,0 +1,49 @@ +{% extends "get_together/base.html" %} +{% load static %} + +{% block content %} +
+
+
+
+

Pick a name and location for your new team

+ +
+ {% csrf_token %} +

{% include "events/new_team_form.html" %}

+

+
+
+
+
+
+{% endblock %} + +{% block javascript %} + + + +{% endblock %} diff --git a/get_together/templates/get_together/places/create_place.html b/get_together/templates/get_together/places/create_place.html index 40c5cd7..ecfc860 100644 --- a/get_together/templates/get_together/places/create_place.html +++ b/get_together/templates/get_together/places/create_place.html @@ -12,7 +12,7 @@ {% include "events/place_form.html" %}

- + Add a Place later diff --git a/get_together/templates/get_together/users/login.html b/get_together/templates/get_together/users/login.html index f7769f7..780c683 100644 --- a/get_together/templates/get_together/users/login.html +++ b/get_together/templates/get_together/users/login.html @@ -2,18 +2,17 @@ {% block content %}
-

Welcome to Get Together!

-

Login

+

Signup or Login to get started

-{% if settings.SOCIAL_AUTH_GOOGLE_OAUTH2_KEY %}Google{% endif %} -{% if settings.SOCIAL_AUTH_FACEBOOK_KEY %}Faceboook{% endif %} -{% if settings.SOCIAL_AUTH_TWITTER_KEY %}Twitter{% endif %} -{% if settings.SOCIAL_AUTH_GITHUB_KEY %}GitHub{% endif %} -{% if settings.DEBUG %}Local{% endif %} +{% if settings.SOCIAL_AUTH_GOOGLE_OAUTH2_KEY %}Google{% endif %} +{% if settings.SOCIAL_AUTH_FACEBOOK_KEY %}Faceboook{% endif %} +{% if settings.SOCIAL_AUTH_TWITTER_KEY %}Twitter{% endif %} +{% if settings.SOCIAL_AUTH_GITHUB_KEY %}GitHub{% endif %} +{% if settings.DEBUG %}Local{% endif %}
{% endblock %} diff --git a/get_together/urls.py b/get_together/urls.py index 13a1858..452a811 100644 --- a/get_together/urls.py +++ b/get_together/urls.py @@ -60,7 +60,8 @@ urlpatterns = [ path('team//+delete/', views.delete_team, name='delete-team'), path('team//events.ics', feeds.TeamEventsCalendar(), name='team-event-ical'), - path('+create-team/', views.create_team, name='create-team'), + path('+create-team/', views.start_new_team, name='create-team'), + path('team//+define/', views.define_new_team, name='define-team'), path('team/+create-event/', views.create_event_team_select, name='create-event-team-select'), path('team//+create-event/', views.create_event, name='create-event'), path('events//+edit/', views.edit_event, name='edit-event'), diff --git a/get_together/views/__init__.py b/get_together/views/__init__.py index c40b729..32cf08c 100644 --- a/get_together/views/__init__.py +++ b/get_together/views/__init__.py @@ -26,6 +26,7 @@ from .events import * from .places import * from .user import * from .new_user import * +from .new_team import * from .utils import * KM_PER_DEGREE_LAT = 110.574 @@ -54,7 +55,7 @@ def home(request, *args, **kwards): else : context['city_search'] = False try: - g = get_geoip(request) + g = location.get_geoip(request) if g.latlng is not None and g.latlng[0] is not None and g.latlng[1] is not None: ll = g.latlng context['geoip_lookup'] = True @@ -76,7 +77,7 @@ def home(request, *args, **kwards): except Exception as err: context['geoip_lookup'] = False - print("Geocoder lookup failed for %s" % client_ip, err) + print("Geocoder lookup failed for %s" % request.META.get('REMOTE_ADDR'), err) traceback.print_exc() #import pdb; pdb.set_trace() @@ -98,6 +99,11 @@ def home(request, *args, **kwards): near_teams = Team.objects.filter(city__latitude__gte=minlat, city__latitude__lte=maxlat, city__longitude__gte=minlng, city__longitude__lte=maxlng) context['near_teams'] = sorted(near_teams, key=lambda team: location.team_distance_from(ll, team)) + +# # If there aren't any teams in the user's geoip area, direct them to start one + if context['geoip_lookup'] and len(near_teams) < 1 and len(near_events) < 1: + messages.add_message(request, messages.INFO, message=_('There are no teams or events yet in your area, be the first to start one!')) + return redirect('create-team') except Exception as err: print("Error looking up nearby teams and events", err) traceback.print_exc() diff --git a/get_together/views/events.py b/get_together/views/events.py index 1df06a7..98f62e4 100644 --- a/get_together/views/events.py +++ b/get_together/views/events.py @@ -111,6 +111,7 @@ def create_event(request, team_id): new_series.save() new_event.series = new_series new_event.save() + messages.add_message(request, messages.SUCCESS, message=_('Your event has been scheduled! Next, find a place for your event.')) return redirect('add-place', new_event.id) else: context = { diff --git a/get_together/views/new_team.py b/get_together/views/new_team.py new file mode 100644 index 0000000..935ec3e --- /dev/null +++ b/get_together/views/new_team.py @@ -0,0 +1,71 @@ +from django.utils.translation import ugettext_lazy as _ + +from django.contrib import messages +from django.contrib.auth import logout as logout_user +from django.contrib.auth.decorators import login_required +from django.shortcuts import render, redirect, get_object_or_404 +from django.http import HttpResponse, JsonResponse + +from events.models.profiles import Organization, Team, UserProfile, Member +from events.models.events import Event, CommonEvent, Place, Attendee +from events.forms import TeamForm, NewTeamForm, TeamDefinitionForm +from events import location + +import datetime +import simplejson + +@login_required +def start_new_team(request, *args, **kwargs): + if request.method == 'GET': + form = NewTeamForm() + g = location.get_geoip(request) + if g.latlng is not None and g.latlng[0] is not None and g.latlng[1] is not None: + city = location.get_nearest_city(g.latlng) + if city: + form.initial={'city': city, 'tz': city.tz} + + context = { + 'team_form': form, + } + return render(request, 'get_together/new_team/start_new_team.html', context) + elif request.method == 'POST': + form = NewTeamForm(request.POST) + if form.is_valid(): + new_team = form.save() + new_team.owner_profile = request.user.profile + new_team.save() + Member.objects.create(team=new_team, user=request.user.profile, role=Member.ADMIN) + return redirect('define-team', team_id=new_team.pk) + else: + context = { + 'team_form': form, + } + return render(request, 'get_together/new_team/start_new_team.html', context) + else: + return redirect('home') + +def define_new_team(request, team_id): + team = get_object_or_404(Team, id=team_id) + if request.method == 'GET': + form = TeamDefinitionForm(instance=team) + + context = { + 'team': team, + 'team_form': form, + } + return render(request, 'get_together/new_team/define_team.html', context) + elif request.method == 'POST': + form = TeamDefinitionForm(request.POST, instance=team) + if form.is_valid(): + form.save() + messages.add_message(request, messages.SUCCESS, message=_('Your new team is ready to go! Now it\'s time to plan your first event.')) + return redirect('create-event', team_id=team.id) + else: + context = { + 'team': team, + 'team_form': form, + } + return render(request, 'get_together/new_team/define_team.html', context) + else: + return redirect('home') +