# Django Imports from django import forms from django.core.exceptions import ValidationError from django.utils.safestring import mark_safe from django.utils import timezone # App Imports from go.models import URL, RegisteredUser # Other Imports from crispy_forms.helper import FormHelper from crispy_forms.layout import Layout, Fieldset, Submit, HTML, Div, Field from crispy_forms.bootstrap import StrictButton, PrependedText, Accordion, AccordionGroup from bootstrap3_datetime.widgets import DateTimePicker import datetime from datetime import date class URLForm(forms.ModelForm): # Prevent redirect loop links def clean_target(self): target = self.cleaned_data.get('target') if self.host in target: raise ValidationError("You can't make a Go link to Go silly!") else: return target # Custom target URL field target = forms.URLField( required=True, label='Long URL (Required)', max_length=1000, widget=forms.URLInput(attrs={ 'placeholder': 'https://yoursite.com/' }) ) # Check to make sure the short url has not been used def unique_short(value): try: URL.objects.get(short__iexact=value) except URL.DoesNotExist: return raise ValidationError('Short url already exists.') # Custom short-url field with validators. short = forms.SlugField( required=False, label='Short URL (Optional)', widget=forms.TextInput(), validators=[unique_short], max_length=20, min_length=3, ) DAY = '1 Day' WEEK = '1 Week' MONTH = '1 Month' CUSTOM = 'Custom Date' NEVER = 'Never' EXPIRATION_CHOICES = ( (DAY, DAY), (WEEK, WEEK), (MONTH, MONTH), (NEVER, NEVER), (CUSTOM, CUSTOM), ) # Add preset expiration choices. expires = forms.ChoiceField( required=True, label='Expiration (Required)', choices=EXPIRATION_CHOICES, initial=NEVER, widget=forms.RadioSelect(), ) def valid_date(value): if value > timezone.now(): return else: raise ValidationError('Date can\'t be before today.') # Add a custom expiration choice. expires_custom = forms.DateTimeField( required = False, label='Custom Date', input_formats=['%m-%d-%Y'], validators=[valid_date], widget=DateTimePicker( options={ "format": "MM-DD-YYYY", "pickTime": False, "defaultDate": True, }, icon_attrs={ "class": "fa fa-calendar", }) ) def __init__(self, *args, **kwargs): # Grab that host info self.host = kwargs.pop('host', None) super(URLForm, self).__init__(*args, **kwargs) # Define the basics for crispy-forms self.helper = FormHelper() self.helper.form_method = 'POST' # Some xtra vars for form css purposes self.helper.form_class = 'form-horizontal' self.helper.label_class = 'col-md-1' self.helper.field_class = 'col-md-6' # The main "layout" defined self.helper.layout = Layout( Fieldset('', ####################### Accordion( # Step 1: Long URL AccordionGroup('Step 1: Long URL', Div( HTML("""

Paste the URL you would like to shorten:


"""), 'target', style="background: rgb(#F6F6F6);"), active=True, template='crispy/accordian-group.html'), # Step 2: Short URL AccordionGroup('Step 2: Short URL', Div( HTML("""

Create a custom Go address:


"""), PrependedText( 'short', 'https://go.gmu.edu/'), style="background: rgb(#F6F6F6);"), active=True, template='crispy/accordian-group.html',), # Step 3: Expiration AccordionGroup('Step 3: URL Expiration', Div( HTML("""

Set when you would like your Go address to expire:


"""), 'expires', 'expires_custom', style="background: rgb(#F6F6F6);"), active=True, template='crispy/accordian-group.html'), # FIN template='crispy/accordian.html'), ####################### HTML("""
"""), StrictButton('Shorten', css_class="btn btn-primary btn-md col-md-4", type='submit'))) class Meta: model = URL fields = ('target',) exclude = ('owner', 'short', 'date_created', 'clicks', 'expires') class SignupForm(forms.ModelForm): def validate_username(username): try: registered = RegisteredUser.objects.get(username=username) raise ValidationError('Username "%s" is already in use.' % username) except RegisteredUser.DoesNotExist: return def clean_username(self): # Prevent hax: (non-staff) Users cannot signup for other users data_username = self.cleaned_data.get("username") if not self.request.user.is_staff: if self.request.user.username not in data_username: raise ValidationError('username', "This is not your NetID!") return data_username username = forms.CharField( required=True, label='Mason NetID (Required)', max_length=30, validators=[validate_username], widget=forms.TextInput(attrs={ }), ) full_name = forms.CharField( required=True, label='Full Name (Required)', max_length=100, widget=forms.TextInput(attrs={ }), ) organization = forms.CharField( required=True, label='Organization (Required)', max_length=100, widget=forms.TextInput(attrs={ }) ) description = forms.CharField( required=False, label='Description (Optional)', max_length=200, widget=forms.Textarea(attrs={ }), ) tos_box = forms.BooleanField( required=True, # Need to add a Terms of Service Page and replace the href below label = mark_safe('Do you accept the Terms of Service?'), ) def __init__(self, request, *args, **kwargs): # Necessary to call request in forms.py, is otherwise restricted to views.py and models.py self.request = request super(SignupForm, self).__init__(*args, **kwargs) self.helper = FormHelper() self.helper.form_class = 'form-horizontal' self.helper.label_class = 'col-md-4' self.helper.field_class = 'col-md-6' self.helper.layout = Layout( Fieldset('', Div( # Place in form fields Div( 'username', 'full_name', 'organization', 'description', 'tos_box', css_class='well'), # Extras at bottom StrictButton('Submit',css_class='btn btn-primary btn-md col-md-4', type='submit'), css_class='col-md-6'))) class Meta: model = RegisteredUser fields = '__all__'