Compare commits
46 Commits
newsletter
...
master
Author | SHA1 | Date | |
---|---|---|---|
![]() |
5a54647577 | ||
![]() |
b2798a6630 | ||
![]() |
61a87a7461 | ||
![]() |
d098531700 | ||
![]() |
faaa66aa8e | ||
![]() |
db60f32c07 | ||
![]() |
25b0385a87 | ||
![]() |
97b1b47d6d | ||
![]() |
86061ae781 | ||
![]() |
986847837b | ||
![]() |
65a4227435 | ||
![]() |
c928ab63cd | ||
![]() |
ea30b13c12 | ||
![]() |
1816bb8386 | ||
![]() |
4a324f52c7 | ||
![]() |
8a3f8a56e5 | ||
![]() |
f5afb1e012 | ||
![]() |
18f48df6a1 | ||
![]() |
94573dc28b | ||
![]() |
b03b106fd8 | ||
![]() |
ea8d8826b6 | ||
![]() |
5bcd86db97 | ||
![]() |
e50439a1ff | ||
![]() |
39b964c4e8 | ||
![]() |
18e4414c44 | ||
![]() |
8fa768fd91 | ||
![]() |
d275c8a0a1 | ||
![]() |
5ec22d5ff9 | ||
![]() |
67e91fc5af | ||
![]() |
5214f806b0 | ||
![]() |
02adc8fe6d | ||
![]() |
ac66d766c5 | ||
![]() |
535857581f | ||
![]() |
7362469204 | ||
![]() |
02551b24b5 | ||
![]() |
6c5bcb0131 | ||
![]() |
acb8668991 | ||
![]() |
a577e47acc | ||
![]() |
3b9f4c33f2 | ||
![]() |
62c84570cc | ||
![]() |
d773cba908 | ||
![]() |
38a25437b6 | ||
![]() |
292e9eda2a | ||
![]() |
65d167a8b8 | ||
![]() |
92fda4831a | ||
![]() |
806718824c |
46
Dockerfile
Normal file
@ -0,0 +1,46 @@
|
||||
# The first instruction is what image we want to base our container on
|
||||
# We Use an official Python runtime as a parent image
|
||||
#FROM python:3.6
|
||||
|
||||
# FROM directive instructing base image to build upon
|
||||
FROM python:3.6.6
|
||||
|
||||
# The enviroment variable ensures that the python output is set straight
|
||||
# to the terminal with out buffering it first
|
||||
#ENV PYTHONUNBUFFERED 1
|
||||
|
||||
# App directory
|
||||
RUN mkdir -p /usr/src/app
|
||||
WORKDIR /usr/src/app
|
||||
|
||||
# Install azure event hub client dependencies
|
||||
#COPY p /usr/src/app/
|
||||
|
||||
# Bundle app source
|
||||
COPY . /usr/src/app
|
||||
|
||||
# COPY startup script into known file location in container
|
||||
COPY start.sh /start.sh
|
||||
|
||||
# create root directory for our project in the container
|
||||
#RUN mkdir /esther_kleinhenz_ba
|
||||
|
||||
# EXPOSE port 8000 to allow communication to/from server
|
||||
#EXPOSE 8000
|
||||
|
||||
RUN set -x \
|
||||
&& buildDeps='curl gcc libc6-dev libsqlite3-dev libssl-dev make xz-utils zlib1g-dev'
|
||||
|
||||
# Install any needed packages specified in requirements.txt
|
||||
RUN pip install -r requirements.txt
|
||||
|
||||
# Set the working directory to /esther_kleinhenz_ba
|
||||
#WORKDIR /esther_kleinhenz_ba
|
||||
|
||||
# CMD specifcies the command to execute to start the server running.
|
||||
CMD ["/start.sh"]
|
||||
# done!
|
||||
|
||||
# Copy the current directory contents into the container at /esther_kleinhenz_ba
|
||||
#ADD . /esther_kleinhenz_ba/
|
||||
|
@ -1,5 +1 @@
|
||||
from __future__ import absolute_import, unicode_literals
|
||||
# This will make sure celery is always imported when
|
||||
# Django starts so that shared_task will use this app.
|
||||
from .celeryapp import app as celery_app
|
||||
__all__ = ['celery_app']
|
@ -1,64 +1,28 @@
|
||||
from django.contrib import admin
|
||||
|
||||
from django.contrib.auth import get_user_model
|
||||
from django.contrib.auth.admin import UserAdmin as BaseUserAdmin
|
||||
from django.contrib.auth.models import User
|
||||
|
||||
from .models import Post, CustomUser
|
||||
from .models import ScheduledReport, ReportRecipient, ScheduledReportGroup
|
||||
from .forms import ScheduledReportForm
|
||||
|
||||
|
||||
#external code customised
|
||||
#import from https://docs.djangoproject.com/en/dev/topics/auth/customizing/#extending-the-existing-user-model
|
||||
#start ---
|
||||
class CustomUserInline(admin.StackedInline):
|
||||
model = CustomUser
|
||||
can_delete = False
|
||||
verbose_name_plural = 'customUsers'
|
||||
|
||||
|
||||
# Define a new User admin
|
||||
class UserAdmin(BaseUserAdmin):
|
||||
inlines = (CustomUserInline, )
|
||||
|
||||
|
||||
# Re-register UserAdmin
|
||||
admin.site.unregister(User)
|
||||
admin.site.register(User, UserAdmin)
|
||||
|
||||
admin.site.register(Post)
|
||||
|
||||
|
||||
class ReportRecipientAdmin(admin.TabularInline):
|
||||
model = ReportRecipient
|
||||
class ScheduledReportAdmin(admin.ModelAdmin):
|
||||
"""
|
||||
List display for Scheduled reports in Django admin
|
||||
"""
|
||||
model = ScheduledReport
|
||||
list_display = ('id', 'get_recipients')
|
||||
inlines = [
|
||||
ReportRecipientAdmin
|
||||
]
|
||||
form = ScheduledReportForm
|
||||
def get_recipients(self, model):
|
||||
recipients = model.reportrecep.all().values_list('email', flat=True)
|
||||
if not recipients:
|
||||
return 'No recipients added'
|
||||
recipient_list = ''
|
||||
for recipient in recipients:
|
||||
recipient_list = recipient_list + recipient + ', '
|
||||
return recipient_list[:-2]
|
||||
get_recipients.short_description = 'Recipients'
|
||||
get_recipients.allow_tags = True
|
||||
class ScheduledReportGroupAdmin(admin.ModelAdmin):
|
||||
"""
|
||||
List display for ScheduledReportGroup Admin
|
||||
"""
|
||||
model = ScheduledReportGroup
|
||||
list_display = ('get_scheduled_report_name','get_report_name')
|
||||
def get_scheduled_report_name(self, model):
|
||||
return model.scheduled_report.subject
|
||||
def get_report_name(self, model):
|
||||
return model.report.name
|
||||
get_scheduled_report_name.short_description = "Scheduled Report Name"
|
||||
get_report_name.short_description = "Report Name"
|
||||
show_change_link = True
|
||||
get_report_name.allow_tags = True
|
||||
admin.site.register(ScheduledReport, ScheduledReportAdmin)
|
||||
admin.site.register(ScheduledReportGroup, ScheduledReportGroupAdmin)
|
||||
#end ---
|
@ -1,13 +0,0 @@
|
||||
from __future__ import absolute_import
|
||||
import os
|
||||
from celery import Celery
|
||||
# set the default Django settings module for the 'celery' program.
|
||||
os.environ.setdefault('DJANGO_SETTINGS_MODULE', 'application.settings')
|
||||
from django.conf import settings
|
||||
|
||||
app = Celery('application')
|
||||
# Using a string here means the worker don't have to serialize
|
||||
# the configuration object to child processes.
|
||||
app.config_from_object('django.conf:settings')
|
||||
# Load task modules from all registered Django app configs.
|
||||
app.autodiscover_tasks(lambda: settings.INSTALLED_APPS)
|
@ -3,45 +3,22 @@ from datetime import datetime
|
||||
|
||||
from .models import Post, CustomUser
|
||||
from django.forms import ModelForm, ValidationError
|
||||
|
||||
from taggit.forms import *
|
||||
from django.contrib.auth.forms import UserCreationForm, UserChangeForm
|
||||
|
||||
from datetime import datetime
|
||||
from croniter import croniter
|
||||
from django.forms import ModelForm, ValidationError
|
||||
from .models import ScheduledReport
|
||||
|
||||
#extended code customised
|
||||
#import from https://tutorial.djangogirls.org/en/django_forms/
|
||||
class PostForm(forms.ModelForm):
|
||||
|
||||
class Meta:
|
||||
model = Post
|
||||
fields = ('title', 'text', 'published_date','tags')
|
||||
|
||||
|
||||
class NewTagForm(forms.ModelForm):
|
||||
m_tags = TagField()
|
||||
tags = TagField()
|
||||
|
||||
class Meta:
|
||||
model = CustomUser
|
||||
fields = ['m_tags']
|
||||
|
||||
|
||||
class ScheduledReportForm(ModelForm):
|
||||
class Meta:
|
||||
model = ScheduledReport
|
||||
fields = ['subject', 'cron_expression']
|
||||
fields = ['subject', 'cron_expression']
|
||||
help_texts = {'cron_expression': 'Scheduled time is considered in UTC'}
|
||||
def clean(self):
|
||||
cleaned_data = super(ScheduledReportForm, self).clean()
|
||||
cron_expression = cleaned_data.get("cron_expression")
|
||||
try:
|
||||
iter = croniter(cron_expression, datetime.now())
|
||||
except:
|
||||
raise ValidationError("Incorrect cron expression:\
|
||||
The information you must include is (in order of appearance):\
|
||||
A number (or list of numbers, or range of numbers), m, representing the minute of the hour\
|
||||
A number (or list of numbers, or range of numbers), h, representing the hour of the day\
|
||||
A number (or list of numbers, or range of numbers), dom, representing the day of the month\
|
||||
A number (or list, or range), or name (or list of names), mon, representing the month of the year\
|
||||
A number (or list, or range), or name (or list of names), dow, representing the day of the week\
|
||||
The asterisks (*) in our entry tell cron that for that unit of time, the job should be run every.\
|
||||
Eg. */5 * * * * cron for executing every 5 mins")
|
||||
return cleaned_data
|
||||
fields = ['tags']
|
||||
|
@ -1,9 +1,10 @@
|
||||
# Generated by Django 2.1 on 2018-08-18 15:34
|
||||
# Generated by Django 2.1.2 on 2018-10-30 10:43
|
||||
|
||||
from django.conf import settings
|
||||
from django.db import migrations, models
|
||||
import django.db.models.deletion
|
||||
import django.utils.timezone
|
||||
import taggit.managers
|
||||
|
||||
|
||||
class Migration(migrations.Migration):
|
||||
@ -12,6 +13,7 @@ class Migration(migrations.Migration):
|
||||
|
||||
dependencies = [
|
||||
migrations.swappable_dependency(settings.AUTH_USER_MODEL),
|
||||
('taggit', '0001_initial'),
|
||||
]
|
||||
|
||||
operations = [
|
||||
@ -19,6 +21,8 @@ class Migration(migrations.Migration):
|
||||
name='CustomUser',
|
||||
fields=[
|
||||
('id', models.AutoField(auto_created=True, primary_key=True, serialize=False, verbose_name='ID')),
|
||||
('tags', taggit.managers.TaggableManager(help_text='A comma-separated list of tags.', through='taggit.TaggedItem', to='taggit.Tag', verbose_name='Tags')),
|
||||
('user', models.OneToOneField(null=True, on_delete=django.db.models.deletion.CASCADE, to=settings.AUTH_USER_MODEL)),
|
||||
],
|
||||
),
|
||||
migrations.CreateModel(
|
||||
@ -30,6 +34,44 @@ class Migration(migrations.Migration):
|
||||
('created_date', models.DateTimeField(default=django.utils.timezone.now)),
|
||||
('published_date', models.DateTimeField(blank=True, null=True)),
|
||||
('author', models.ForeignKey(on_delete=django.db.models.deletion.CASCADE, to=settings.AUTH_USER_MODEL)),
|
||||
('tags', taggit.managers.TaggableManager(help_text='A comma-separated list of tags.', through='taggit.TaggedItem', to='taggit.Tag', verbose_name='Tags')),
|
||||
],
|
||||
),
|
||||
migrations.CreateModel(
|
||||
name='Report',
|
||||
fields=[
|
||||
('id', models.AutoField(auto_created=True, primary_key=True, serialize=False, verbose_name='ID')),
|
||||
('report_text', models.TextField()),
|
||||
],
|
||||
),
|
||||
migrations.CreateModel(
|
||||
name='ReportRecipient',
|
||||
fields=[
|
||||
('id', models.AutoField(auto_created=True, primary_key=True, serialize=False, verbose_name='ID')),
|
||||
('email', models.EmailField(max_length=254)),
|
||||
],
|
||||
),
|
||||
migrations.CreateModel(
|
||||
name='ScheduledReport',
|
||||
fields=[
|
||||
('id', models.AutoField(auto_created=True, primary_key=True, serialize=False, verbose_name='ID')),
|
||||
('subject', models.CharField(max_length=200)),
|
||||
('last_run_at', models.DateTimeField(blank=True, null=True)),
|
||||
('next_run_at', models.DateTimeField(blank=True, null=True)),
|
||||
('cron_expression', models.CharField(max_length=200)),
|
||||
],
|
||||
),
|
||||
migrations.CreateModel(
|
||||
name='ScheduledReportGroup',
|
||||
fields=[
|
||||
('id', models.AutoField(auto_created=True, primary_key=True, serialize=False, verbose_name='ID')),
|
||||
('report', models.ForeignKey(on_delete=django.db.models.deletion.CASCADE, related_name='report', to='application.Report')),
|
||||
('scheduled_report', models.ForeignKey(on_delete=django.db.models.deletion.CASCADE, related_name='relatedscheduledreport', to='application.ScheduledReport')),
|
||||
],
|
||||
),
|
||||
migrations.AddField(
|
||||
model_name='reportrecipient',
|
||||
name='scheduled_report',
|
||||
field=models.ForeignKey(on_delete=django.db.models.deletion.CASCADE, related_name='reportrecep', to='application.ScheduledReport'),
|
||||
),
|
||||
]
|
||||
|
@ -1,35 +0,0 @@
|
||||
# Generated by Django 2.1 on 2018-08-18 15:34
|
||||
|
||||
from django.conf import settings
|
||||
from django.db import migrations, models
|
||||
import django.db.models.deletion
|
||||
import taggit.managers
|
||||
|
||||
|
||||
class Migration(migrations.Migration):
|
||||
|
||||
initial = True
|
||||
|
||||
dependencies = [
|
||||
('taggit', '0001_initial'),
|
||||
('application', '0001_initial'),
|
||||
migrations.swappable_dependency(settings.AUTH_USER_MODEL),
|
||||
]
|
||||
|
||||
operations = [
|
||||
migrations.AddField(
|
||||
model_name='post',
|
||||
name='tags',
|
||||
field=taggit.managers.TaggableManager(help_text='A comma-separated list of tags.', through='taggit.TaggedItem', to='taggit.Tag', verbose_name='Tags'),
|
||||
),
|
||||
migrations.AddField(
|
||||
model_name='customuser',
|
||||
name='mytags',
|
||||
field=taggit.managers.TaggableManager(help_text='A comma-separated list of tags.', through='taggit.TaggedItem', to='taggit.Tag', verbose_name='Tags'),
|
||||
),
|
||||
migrations.AddField(
|
||||
model_name='customuser',
|
||||
name='user',
|
||||
field=models.OneToOneField(on_delete=django.db.models.deletion.CASCADE, to=settings.AUTH_USER_MODEL),
|
||||
),
|
||||
]
|
37
application/migrations/0002_auto_20181030_1223.py
Normal file
@ -0,0 +1,37 @@
|
||||
# Generated by Django 2.1.2 on 2018-10-30 11:23
|
||||
|
||||
from django.db import migrations
|
||||
|
||||
|
||||
class Migration(migrations.Migration):
|
||||
|
||||
dependencies = [
|
||||
('application', '0001_initial'),
|
||||
]
|
||||
|
||||
operations = [
|
||||
migrations.RemoveField(
|
||||
model_name='reportrecipient',
|
||||
name='scheduled_report',
|
||||
),
|
||||
migrations.RemoveField(
|
||||
model_name='scheduledreportgroup',
|
||||
name='report',
|
||||
),
|
||||
migrations.RemoveField(
|
||||
model_name='scheduledreportgroup',
|
||||
name='scheduled_report',
|
||||
),
|
||||
migrations.DeleteModel(
|
||||
name='Report',
|
||||
),
|
||||
migrations.DeleteModel(
|
||||
name='ReportRecipient',
|
||||
),
|
||||
migrations.DeleteModel(
|
||||
name='ScheduledReport',
|
||||
),
|
||||
migrations.DeleteModel(
|
||||
name='ScheduledReportGroup',
|
||||
),
|
||||
]
|
@ -1,20 +0,0 @@
|
||||
# Generated by Django 2.1 on 2018-08-30 16:14
|
||||
|
||||
from django.conf import settings
|
||||
from django.db import migrations, models
|
||||
import django.db.models.deletion
|
||||
|
||||
|
||||
class Migration(migrations.Migration):
|
||||
|
||||
dependencies = [
|
||||
('application', '0002_auto_20180818_1734'),
|
||||
]
|
||||
|
||||
operations = [
|
||||
migrations.AlterField(
|
||||
model_name='customuser',
|
||||
name='user',
|
||||
field=models.OneToOneField(default=None, null=True, on_delete=django.db.models.deletion.CASCADE, to=settings.AUTH_USER_MODEL),
|
||||
),
|
||||
]
|
19
application/migrations/0003_auto_20181030_1633.py
Normal file
@ -0,0 +1,19 @@
|
||||
# Generated by Django 2.1.2 on 2018-10-30 15:33
|
||||
|
||||
from django.db import migrations
|
||||
import taggit.managers
|
||||
|
||||
|
||||
class Migration(migrations.Migration):
|
||||
|
||||
dependencies = [
|
||||
('application', '0002_auto_20181030_1223'),
|
||||
]
|
||||
|
||||
operations = [
|
||||
migrations.AlterField(
|
||||
model_name='customuser',
|
||||
name='tags',
|
||||
field=taggit.managers.TaggableManager(blank=True, help_text='A comma-separated list of tags.', through='taggit.TaggedItem', to='taggit.Tag', verbose_name='Tags'),
|
||||
),
|
||||
]
|
@ -1,20 +0,0 @@
|
||||
# Generated by Django 2.1 on 2018-08-31 08:19
|
||||
|
||||
from django.conf import settings
|
||||
from django.db import migrations, models
|
||||
import django.db.models.deletion
|
||||
|
||||
|
||||
class Migration(migrations.Migration):
|
||||
|
||||
dependencies = [
|
||||
('application', '0003_auto_20180830_1814'),
|
||||
]
|
||||
|
||||
operations = [
|
||||
migrations.AlterField(
|
||||
model_name='customuser',
|
||||
name='user',
|
||||
field=models.OneToOneField(null=True, on_delete=django.db.models.deletion.CASCADE, to=settings.AUTH_USER_MODEL),
|
||||
),
|
||||
]
|
@ -1,18 +0,0 @@
|
||||
# Generated by Django 2.1 on 2018-10-19 14:45
|
||||
|
||||
from django.db import migrations
|
||||
|
||||
|
||||
class Migration(migrations.Migration):
|
||||
|
||||
dependencies = [
|
||||
('application', '0004_auto_20180831_1019'),
|
||||
]
|
||||
|
||||
operations = [
|
||||
migrations.RenameField(
|
||||
model_name='customuser',
|
||||
old_name='mytags',
|
||||
new_name='tags',
|
||||
),
|
||||
]
|
@ -1,51 +0,0 @@
|
||||
# Generated by Django 2.1 on 2018-10-21 11:47
|
||||
|
||||
from django.db import migrations, models
|
||||
import django.db.models.deletion
|
||||
|
||||
|
||||
class Migration(migrations.Migration):
|
||||
|
||||
dependencies = [
|
||||
('application', '0005_auto_20181019_1645'),
|
||||
]
|
||||
|
||||
operations = [
|
||||
migrations.CreateModel(
|
||||
name='Report',
|
||||
fields=[
|
||||
('id', models.AutoField(auto_created=True, primary_key=True, serialize=False, verbose_name='ID')),
|
||||
('report_text', models.TextField()),
|
||||
],
|
||||
),
|
||||
migrations.CreateModel(
|
||||
name='ReportRecipient',
|
||||
fields=[
|
||||
('id', models.AutoField(auto_created=True, primary_key=True, serialize=False, verbose_name='ID')),
|
||||
('email', models.EmailField(max_length=254)),
|
||||
],
|
||||
),
|
||||
migrations.CreateModel(
|
||||
name='ScheduledReport',
|
||||
fields=[
|
||||
('id', models.AutoField(auto_created=True, primary_key=True, serialize=False, verbose_name='ID')),
|
||||
('subject', models.CharField(max_length=200)),
|
||||
('last_run_at', models.DateTimeField(blank=True, null=True)),
|
||||
('next_run_at', models.DateTimeField(blank=True, null=True)),
|
||||
('cron_expression', models.CharField(max_length=200)),
|
||||
],
|
||||
),
|
||||
migrations.CreateModel(
|
||||
name='ScheduledReportGroup',
|
||||
fields=[
|
||||
('id', models.AutoField(auto_created=True, primary_key=True, serialize=False, verbose_name='ID')),
|
||||
('report', models.ForeignKey(on_delete=django.db.models.deletion.CASCADE, related_name='report', to='application.Report')),
|
||||
('scheduled_report', models.ForeignKey(on_delete=django.db.models.deletion.CASCADE, related_name='relatedscheduledreport', to='application.ScheduledReport')),
|
||||
],
|
||||
),
|
||||
migrations.AddField(
|
||||
model_name='reportrecipient',
|
||||
name='scheduled_report',
|
||||
field=models.ForeignKey(on_delete=django.db.models.deletion.CASCADE, related_name='reportrecep', to='application.ScheduledReport'),
|
||||
),
|
||||
]
|
@ -1,24 +1,29 @@
|
||||
from django.db import models
|
||||
|
||||
from django.contrib.auth.models import User
|
||||
from django.utils import timezone
|
||||
|
||||
from taggit.managers import TaggableManager
|
||||
from datetime import datetime
|
||||
from croniter import croniter
|
||||
|
||||
|
||||
|
||||
#external code customised
|
||||
#import from https://docs.djangoproject.com/en/dev/topics/auth/customizing/#extending-the-existing-user-model
|
||||
class CustomUser(models.Model):
|
||||
user = models.OneToOneField(User, null=True, on_delete=models.CASCADE)
|
||||
tags = TaggableManager()
|
||||
#add tags to User Model with possibility to leave empty
|
||||
tags = TaggableManager(blank=True)
|
||||
|
||||
|
||||
#external code customised
|
||||
#import from https://tutorial.djangogirls.org/en/django_models/
|
||||
class Post(models.Model):
|
||||
author = models.ForeignKey('auth.User', on_delete=models.CASCADE)
|
||||
title = models.CharField(max_length=200)
|
||||
text = models.TextField()
|
||||
created_date = models.DateTimeField(
|
||||
default=timezone.now)
|
||||
published_date = models.DateTimeField(
|
||||
blank=True, null=True)
|
||||
created_date = models.DateTimeField(default=timezone.now)
|
||||
published_date = models.DateTimeField(blank=True, null=True)
|
||||
#add tags to Post Model
|
||||
tags = TaggableManager()
|
||||
|
||||
def publish(self):
|
||||
@ -27,39 +32,3 @@ class Post(models.Model):
|
||||
|
||||
def __str__(self):
|
||||
return self.title
|
||||
|
||||
class Report(models.Model):
|
||||
report_text = models.TextField()
|
||||
|
||||
class ScheduledReport(models.Model):
|
||||
"""
|
||||
Contains email subject and cron expression,to evaluate when the email has to be sent
|
||||
"""
|
||||
subject = models.CharField(max_length=200)
|
||||
last_run_at = models.DateTimeField(null=True, blank=True)
|
||||
next_run_at = models.DateTimeField(null=True, blank=True)
|
||||
cron_expression = models.CharField(max_length=200)
|
||||
def save(self, *args, **kwargs):
|
||||
"""
|
||||
function to evaluate "next_run_at" using the cron expression, so that it is updated once the report is sent.
|
||||
"""
|
||||
self.last_run_at = datetime.now()
|
||||
iter = croniter(self.cron_expression, self.last_run_at)
|
||||
self.next_run_at = iter.get_next(datetime)
|
||||
super(ScheduledReport, self).save(*args, **kwargs)
|
||||
def __unicode__(self):
|
||||
return self.subject
|
||||
|
||||
class ScheduledReportGroup(models.Model):
|
||||
"""
|
||||
Many to many mapping between reports which will be sent out in a scheduled report
|
||||
"""
|
||||
report = models.ForeignKey(Report, related_name='report', on_delete=models.CASCADE)
|
||||
scheduled_report = models.ForeignKey(ScheduledReport,
|
||||
related_name='relatedscheduledreport', on_delete=models.CASCADE)
|
||||
class ReportRecipient(models.Model):
|
||||
"""
|
||||
Stores all the recipients of the given scheduled report
|
||||
"""
|
||||
email = models.EmailField()
|
||||
scheduled_report = models.ForeignKey(ScheduledReport, related_name='reportrecep', on_delete=models.CASCADE)
|
@ -1,7 +0,0 @@
|
||||
from celery.task.schedules import crontab
|
||||
from celery.decorators import periodic_task
|
||||
from .email_service import send_emails
|
||||
# this will run every minute, see http://celeryproject.org/docs/reference/celery.task.schedules.html#celery.task.schedules.crontab
|
||||
@periodic_task(run_every=crontab(hour="*", minute="*", day_of_week="*"))
|
||||
def trigger_emails():
|
||||
send_emails()
|
@ -20,8 +20,8 @@
|
||||
<img src="{% static 'images/efi.jpg' %}" width="35" height="35" alt="efi">
|
||||
</a>
|
||||
|
||||
<button class="navbar-toggler" type="button" data-toggle="collapse" data-target="#navbarSupportedContent" aria-controls="navbarSupportedContent"
|
||||
aria-expanded="false" aria-label="Toggle navigation">
|
||||
<button class="navbar-toggler" type="button" data-toggle="collapse" data-target="#navbarSupportedContent"
|
||||
aria-controls="navbarSupportedContent" aria-expanded="false" aria-label="Toggle navigation">
|
||||
<span class="navbar-toggler-icon"></span>
|
||||
</button>
|
||||
|
||||
@ -32,13 +32,16 @@
|
||||
<a class="nav-link" href="{% url 'admin:index' %}">Administration</a>
|
||||
</li>
|
||||
<li>
|
||||
<a class="nav-link" href="{% url 'post_list' %}">Dashboard</a>
|
||||
<a class="nav-link" href="{% url 'tag_list' %}">Dashboard</a>
|
||||
</li>
|
||||
<li class="nav-item">
|
||||
<a class="nav-link" href="{% url 'student_page' %}">Studentenansicht</a>
|
||||
<a class="nav-link" href="{% url 'search_add' %}">Suche</a>
|
||||
</li>
|
||||
<li class="nav-item">
|
||||
<a class="nav-link" href="{% url 'blog_search_list_view' %}">Suche</a>
|
||||
<a class="nav-link" href="{% url 'post_new' %}" class="top-menu"><span class="glyphicon glyphicon-plus">+</span></a>
|
||||
</li>
|
||||
<li class="nav-item">
|
||||
<a class="nav-link" href="{% url 'post_draft_list' %}" class="top-menu"><span class="glyphicon glyphicon-edit">Entwürfe</span></a>
|
||||
</li>
|
||||
{% elif user.is_staff %}
|
||||
<li class="nav-item">
|
||||
@ -46,11 +49,23 @@
|
||||
</li>
|
||||
|
||||
<li class="nav-item">
|
||||
<a class="nav-link" href="{% url 'post_list' %}">Dashboard</a>
|
||||
<a class="nav-link" href="{% url 'tag_list' %}">Dashboard</a>
|
||||
</li>
|
||||
{% else %}
|
||||
<li class="nav-item">
|
||||
<a class="nav-link" href="{% url 'student_page' %}">{{user.get_username}}'s Dashboard</a>
|
||||
<a class="nav-link" href="{% url 'search_add' %}">Suche</a>
|
||||
</li>
|
||||
<li class="nav-item">
|
||||
<a class="nav-link" href="{% url 'post_new' %}" class="top-menu"><span class="glyphicon glyphicon-plus">+</span></a>
|
||||
</li>
|
||||
<li class="nav-item">
|
||||
<a class="nav-link" href="{% url 'post_draft_list' %}" class="top-menu"><span class="glyphicon glyphicon-edit">Entwürfe</span></a>
|
||||
</li>
|
||||
{% elif user.is_authenticated %}
|
||||
<li class="nav-item">
|
||||
<a class="nav-link" href="{% url 'tag_list' %}">{{user.get_username}}'s Dashboard</a>
|
||||
</li>
|
||||
<li class="nav-item">
|
||||
<a class="nav-link" href="{% url 'search_add' %}">Suche</a>
|
||||
</li>
|
||||
{% endif %}
|
||||
</ul>
|
||||
@ -71,16 +86,19 @@
|
||||
</a>
|
||||
<ul id="login-dp" class="dropdown-menu">
|
||||
<li>
|
||||
<form class="form" role="form" method="post" action="{% url 'login' %}" accept-charset="UTF-8" id="login-nav">
|
||||
<form class="form" role="form" method="post" action="{% url 'login' %}" accept-charset="UTF-8"
|
||||
id="login-nav">
|
||||
{% csrf_token %} {% if next %}
|
||||
<input type="hidden" name="next" value="{{ next }}" /> {% endif %}
|
||||
<div class="form-group">
|
||||
<label class="sr-only" for="username">Benutzername</label>
|
||||
<input type="text" class="form-control" id="username" name="username" placeholder="Benutzername" required>
|
||||
<input type="text" class="form-control" id="username" name="username"
|
||||
placeholder="Benutzername" required>
|
||||
</div>
|
||||
<div class="form-group">
|
||||
<label class="sr-only" for="password">Kennwort</label>
|
||||
<input type="password" class="form-control" id="password" name="password" placeholder="Kennwort" required>
|
||||
<input type="password" class="form-control" id="password" name="password"
|
||||
placeholder="Kennwort" required>
|
||||
</div>
|
||||
<div class="form-group">
|
||||
<button type="submit" class="btn btn-primary btn-block">Anmelden</button>
|
||||
@ -94,9 +112,10 @@
|
||||
</div>
|
||||
</nav>
|
||||
|
||||
<!-- Content -->
|
||||
<div class="content container">
|
||||
<div class="row">
|
||||
<div class="col-md-8">
|
||||
<div class="col-md-12">
|
||||
{% block content %} {% endblock %}
|
||||
</div>
|
||||
</div>
|
||||
@ -107,5 +126,4 @@
|
||||
<script src="{% static 'js/app.js' %}"></script>
|
||||
<script src="{% static 'bootstrap/js/bootstrap.bundle.js' %}"></script>
|
||||
</body>
|
||||
|
||||
</html>
|
@ -1,27 +0,0 @@
|
||||
{% extends "base.html" %} {% block content %}
|
||||
{% load taggit_templatetags2_tags %}
|
||||
{% get_tagcloud as tags for 'application' %}
|
||||
{% include_tagcloud 'application.Post' %}
|
||||
<form type="get" action="." style="margin: 0">
|
||||
<input id="search_box" type="text" name="search_box" placeholder="Search...">
|
||||
<button id="search_submit" type="submit">Suchen</button>
|
||||
</form>
|
||||
<div>
|
||||
{% for post in posts %}
|
||||
<div class="post">
|
||||
<div class="date">
|
||||
{{ post.published_date }}
|
||||
</div>
|
||||
<h1>
|
||||
<a href="{% url 'post_detail' pk=post.pk %}">{{ post.title }}</a>
|
||||
</h1>
|
||||
<p>{{ post.text|linebreaks }}</p>
|
||||
Tags: {% for tag in post.tags.all %}
|
||||
<a href="{% url 'post_list_by_tag' tag.slug %}">{{ tag.name }}</a>
|
||||
{% if not forloop.last %}, {% endif %} {% endfor %} <p>
|
||||
{{ post.author }}
|
||||
</p>
|
||||
</div>
|
||||
{% endfor %}
|
||||
</div>
|
||||
{% endblock %}
|
@ -1,4 +1,12 @@
|
||||
<!--
|
||||
marked parts of the template is external code
|
||||
import from https://tutorial.djangogirls.org/en/template_extending/
|
||||
-->
|
||||
{% extends 'base.html' %}{% block content %}
|
||||
{% load hitcount_tags %}
|
||||
<div class="col-md-8">
|
||||
|
||||
<!-- imported start -->
|
||||
<div class="post">
|
||||
{% if post.published_date %}
|
||||
<div class="date">
|
||||
@ -6,23 +14,31 @@
|
||||
</div>
|
||||
{% else %}
|
||||
<a class="btn btn-outline-dark" href="{% url 'post_publish' pk=post.pk %}">Publish</a>
|
||||
<!-- imported end -->
|
||||
|
||||
{% endif %} {% if user.is_staff and user == post.author %}
|
||||
|
||||
<!-- imported start -->
|
||||
<a class="btn btn-outline-dark" href="{% url 'post_edit' pk=post.pk %}">
|
||||
<span class="glyphicon glyphicon-pencil">Edit</span>
|
||||
<span class="glyphicon glyphicon-pencil">Bearbeiten</span>
|
||||
</a>
|
||||
<a class="btn btn-outline-dark" href="{% url 'post_remove' pk=post.pk %}">
|
||||
<span class="glyphicon glyphicon-remove">Remove</span>
|
||||
<span class="glyphicon glyphicon-remove">Löschen</span>
|
||||
</a>
|
||||
{% endif %}
|
||||
<h1>{{ post.title }}</h1>
|
||||
<p>{{ post.text|linebreaksbr }}</p>
|
||||
<!-- imported end -->
|
||||
Tags: {% for tag in post.tags.all %}
|
||||
<a href="{% url 'post_list_by_tag' tag.slug %}">{{ tag.name }}</a>
|
||||
{% if not forloop.last %}, {% endif %} {% endfor %} <p>
|
||||
{{ post.author }}
|
||||
{% if user.is_staff %}
|
||||
<p>
|
||||
{% for tag in post.tags.all %}
|
||||
<a href="{% url 'blog_search_list_view' %}">{{ tag.name }}, </a>
|
||||
{% endfor %}
|
||||
{% get_hit_count for post %} Benutzer haben diesen Post bereits gelesen!
|
||||
</p>
|
||||
<p>
|
||||
Autor: {{ post.author }}
|
||||
{% endif %}
|
||||
</p>
|
||||
</div>
|
||||
</div>
|
||||
{% endblock %}
|
@ -1,11 +1,20 @@
|
||||
<!--
|
||||
external code
|
||||
import from https://tutorial.djangogirls.org/en/template_extending/
|
||||
-->
|
||||
{% extends 'base.html' %}
|
||||
|
||||
{% block content %}
|
||||
<div class="row">
|
||||
<div class="col-md-8">
|
||||
<h1> Unveröffentliche Artikel<br>
|
||||
</h1>
|
||||
{% for post in posts %}
|
||||
<div class="post">
|
||||
<div class="post mt-5">
|
||||
<p class="date">created: {{ post.created_date|date:'d-m-Y' }}</p>
|
||||
<h1><a href="{% url 'post_detail' pk=post.pk %}">{{ post.title }}</a></h1>
|
||||
<p>{{ post.text|truncatechars:200 }}</p>
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
{% endfor %}
|
||||
{% endblock %}
|
@ -1,9 +1,16 @@
|
||||
<!--
|
||||
external code
|
||||
import from https://tutorial.djangogirls.org/en/template_extending/
|
||||
-->
|
||||
{% extends 'base.html' %}
|
||||
|
||||
{% block content %}
|
||||
<h1>New post</h1>
|
||||
<form method="POST" class="post-form">{% csrf_token %}
|
||||
<div class="row">
|
||||
<div class="mx-auto">
|
||||
<h1>Artikel-Editor</h1>
|
||||
<form method="POST" class="mt-5 post-form">{% csrf_token %}
|
||||
{{ form.as_p }}
|
||||
<button type="submit" class="save btn btn-outline-dark">Save</button>
|
||||
<button type="submit" class="save btn btn-outline-dark">Sichern</button>
|
||||
</form>
|
||||
</div>
|
||||
</div>
|
||||
{% endblock %}
|
@ -1,14 +1,43 @@
|
||||
<!--
|
||||
marked parts of the template is external code
|
||||
import from https://tutorial.djangogirls.org/en/template_extending/
|
||||
-->
|
||||
{% extends 'base.html' %} {% block content %}
|
||||
{% if tag %} Posts tagged with "{{ tag.name }}" {% endif %}
|
||||
<div class="row">
|
||||
{% if tag %} <div class="mx-auto">
|
||||
<h2> Artikel mit dem Tag "{{ tag.name }}" </h2>
|
||||
</div>{% endif %}
|
||||
</div>
|
||||
<!-- imported start -->
|
||||
<div class="row">
|
||||
<div class="mx-auto">
|
||||
<h1> Artikleübersicht </h1>
|
||||
</div>
|
||||
</div>
|
||||
<!-- imported end -->
|
||||
|
||||
<div class="row">
|
||||
<div class="col-md-8">
|
||||
{% if messages %}
|
||||
<div class="messages">
|
||||
{% for message in messages %}
|
||||
<p {% if message.tags %} class="{{ message.tags }}" {% endif %}>{{ message }}</p>
|
||||
{% endfor %}
|
||||
</div>
|
||||
|
||||
<!-- imported start -->
|
||||
{% endif %}
|
||||
{% for post in posts %}
|
||||
<div class="post">
|
||||
<div class="mt-5 post">
|
||||
<div class="date">
|
||||
{{ post.published_date }}
|
||||
</div>
|
||||
<h1>
|
||||
<h2>
|
||||
<a href="{% url 'post_detail' pk=post.pk %}">{{ post.title }}</a>
|
||||
</h1>
|
||||
</h2>
|
||||
<p>{{ post.text|linebreaks }}</p>
|
||||
<!-- imported end -->
|
||||
|
||||
Tags: {% for tag in post.tags.all %}
|
||||
<a href="{% url 'post_list_by_tag' tag.slug %}">{{ tag.name }}</a>
|
||||
{% if not forloop.last %}, {% endif %} {% endfor %} <p>
|
||||
@ -17,4 +46,5 @@
|
||||
</div>
|
||||
{% endfor %}
|
||||
</div>
|
||||
</div>
|
||||
{% endblock %}
|
@ -28,5 +28,28 @@
|
||||
<input type="hidden" name="next" value="{{ next }}" />
|
||||
</div>
|
||||
</form>
|
||||
|
||||
<div>
|
||||
<div class="row">
|
||||
<div class="col-md-8">
|
||||
<div>
|
||||
{% for post in posts %}
|
||||
<div class="mt-5 post">
|
||||
<div class="date">
|
||||
{{ post.published_date }}
|
||||
</div>
|
||||
<h2>
|
||||
<a href="{% url 'post_detail' pk=post.pk %}">{{ post.title }}</a>
|
||||
</h2>
|
||||
<p>{{ post.text|linebreaks }}</p>
|
||||
Tags: {% for tag in post.tags.all %}
|
||||
<a href="{% url 'post_list_by_tag' tag.slug %}">{{ tag.name }}</a>
|
||||
{% if not forloop.last %}, {% endif %} {% endfor %} <p>
|
||||
{{ post.author }}
|
||||
</p>
|
||||
</div>
|
||||
{% endfor %}
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
{% endblock %}
|
86
application/templates/search_add.html
Normal file
@ -0,0 +1,86 @@
|
||||
<!--
|
||||
marked parts of the template is external code
|
||||
import from https://github.com/fizista/django-taggit-templatetags2
|
||||
-->
|
||||
{% extends "base.html" %} {% block content %}
|
||||
|
||||
<!-- imported start -->
|
||||
{% load taggit_templatetags2_tags %}
|
||||
<div class="container">
|
||||
<div class="row equal">
|
||||
<div class="col-md-6">
|
||||
<h1> Alle Tags im Überblick<br>
|
||||
</h1>
|
||||
<div class="col-md-8" style="zoom: 1.6">
|
||||
<div style="text-align:center;padding:5px;">
|
||||
{% get_tagcloud as tags for 'application' %}
|
||||
{% include_tagcloud 'application.Post' %}
|
||||
</div>
|
||||
</div>
|
||||
<!-- imported end -->
|
||||
|
||||
<div style="padding-bottom:40px;">
|
||||
</div>
|
||||
Tag-Suche
|
||||
<div id="custom-search-input">
|
||||
<form type="get" action="." style="margin: 0" class="post-form">
|
||||
<p>
|
||||
<input id="search_box" type="text" name="search_box" placeholder="Search...">
|
||||
</p>
|
||||
<span>
|
||||
<button id="search_submit" type="submit" class="save btn btn-outline-dark">Suchen</button>
|
||||
</button>
|
||||
</span>
|
||||
</form>
|
||||
<div class="pt-4">
|
||||
{% for post in posts %}
|
||||
<div class="post">
|
||||
<div class="date">
|
||||
{{ post.published_date }}
|
||||
</div>
|
||||
<h1>
|
||||
<a href="{% url 'post_detail' pk=post.pk %}">{{ post.title }}</a>
|
||||
</h1>
|
||||
<p>{{ post.text|linebreaks }}</p>
|
||||
{% for tag in post.tags.all %}
|
||||
<a href="{% url 'post_list_by_tag' tag.slug %}">{{ tag.name }}</a>
|
||||
{% if not forloop.last %}, {% endif %} {% endfor %} <p>
|
||||
{{ post.author }}
|
||||
</p>
|
||||
</div>
|
||||
{% endfor %}
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
<div class="col-md-6">
|
||||
<h1> Bereits abonnierten Tags
|
||||
</h1>
|
||||
<div class="col-md-10">
|
||||
{% if tags %}
|
||||
<div style="text-align:center;padding-top:40px;">
|
||||
{% for tag in tagsuser %}
|
||||
<a class="text-white bg-dark p-3" href="{% url 'post_list_by_tag' tag.slug %}">{{ tag.name }}</a>
|
||||
{% endfor %}
|
||||
</div>
|
||||
{% endif %}
|
||||
</div>
|
||||
<div style="padding-bottom:145px;">
|
||||
</div>
|
||||
<div class="mx-auto">
|
||||
<div>
|
||||
Abonniere deine Tags hier!
|
||||
</div>
|
||||
<form class="post-form" method="post">
|
||||
{% csrf_token %} {{form.as_p}}
|
||||
<button type="submit" class="save btn btn-outline-dark">Sichern</button>
|
||||
</form>
|
||||
{% if messages %}
|
||||
{% for message in messages %}
|
||||
<div {% if message.tags %} class="{{ message.tags }}" {% endif %}>{{ message }}</div>
|
||||
{% endfor %}
|
||||
{% endif %}
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
{% endblock %}
|
@ -1,24 +0,0 @@
|
||||
{% extends "base.html" %} {% block content %} {% load taggit_templatetags2_tags %} {% get_taglist as tags for 'application.post'%}
|
||||
|
||||
<div id="">
|
||||
<ul>
|
||||
{% for tag in tags %}
|
||||
<li>{{tag}}
|
||||
<a class="btn btn-outline-dark" href="{% url 'tag_remove' tag.slug %}">
|
||||
<span class="glyphicon glyphicon-remove">Remove</span>
|
||||
</a>
|
||||
</li>
|
||||
{{ result }} {% endfor %}
|
||||
</ul>
|
||||
</div>
|
||||
<div>
|
||||
{% if messages %}
|
||||
{% for message in messages %}
|
||||
<div {% if message.tags %} class="{{ message.tags }}" {% endif %}>{{ message }}</div>
|
||||
{% endfor %}
|
||||
{% endif %}
|
||||
<form class="post-form" method="post">
|
||||
{% csrf_token %} {{form.as_p}}
|
||||
<button type="submit" class="save btn btn-outline-dark">Save</button>
|
||||
</form>
|
||||
{% endblock %}
|
@ -1,17 +1,31 @@
|
||||
<!--
|
||||
marked parts of the template is external code
|
||||
import from https://tutorial.djangogirls.org/en/template_extending/
|
||||
-->
|
||||
{% extends "base.html" %} {% block content %}
|
||||
{% load taggit_templatetags2_tags %}
|
||||
|
||||
<div>
|
||||
<div class="row">
|
||||
<div class="mx-auto">
|
||||
<h1>
|
||||
Dein Newsfeed ...
|
||||
</h1>
|
||||
</div>
|
||||
</div>
|
||||
<div class="row">
|
||||
<div class="col-md-8">
|
||||
<div>
|
||||
|
||||
<!-- imported start -->
|
||||
{% for post in posts %}
|
||||
<div class="post">
|
||||
<div class="mt-5 post">
|
||||
<div class="date">
|
||||
{{ post.published_date }}
|
||||
</div>
|
||||
<h1>
|
||||
<h2>
|
||||
<a href="{% url 'post_detail' pk=post.pk %}">{{ post.title }}</a>
|
||||
</h1>
|
||||
</h2>
|
||||
<!-- imported end -->
|
||||
|
||||
<p>{{ post.text|linebreaks }}</p>
|
||||
Tags: {% for tag in post.tags.all %}
|
||||
<a href="{% url 'post_list_by_tag' tag.slug %}">{{ tag.name }}</a>
|
||||
@ -19,7 +33,14 @@
|
||||
{{ post.author }}
|
||||
</p>
|
||||
</div>
|
||||
|
||||
{% endfor %}
|
||||
</div>
|
||||
</div>
|
||||
<div class="mt-5 col-md-4">
|
||||
<div class="float-right">
|
||||
<a id="search_submit" href="/posts" class="save btn btn-outline-dark">alle Artikel
|
||||
</a>
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
{% endblock %}
|
@ -1,20 +0,0 @@
|
||||
{% extends 'base.html' %} {% block content %}
|
||||
{% if tag %} Posts tagged with "{{ tag.name }}" {% endif %}
|
||||
{% for post in posts %}
|
||||
<div class="post">
|
||||
<div class="date">
|
||||
{{ post.published_date }}
|
||||
</div>
|
||||
<h1>
|
||||
<a href="{% url 'post_detail' pk=post.pk %}">{{ post.title }}</a>
|
||||
</h1>
|
||||
<p>{{ post.text|linebreaks }}</p>
|
||||
Tags: {% for tag in post.tags.all %}
|
||||
<a href="{% url 'student_page_remove' tag.slug %}">{{ tag.name }}</a>
|
||||
{% if not forloop.last %}, {% endif %} {% endfor %} <p>
|
||||
{{ post.author }}
|
||||
</p>
|
||||
</div>
|
||||
{% endfor %}
|
||||
</div>
|
||||
{% endblock %}
|
@ -2,28 +2,29 @@ from django.conf import settings
|
||||
from django.conf.urls import url
|
||||
from django.conf.urls import include
|
||||
from django.urls import path
|
||||
from taggit_templatetags2 import urls as taggit_templatetags2_urls
|
||||
|
||||
from . import views
|
||||
|
||||
from taggit_templatetags2 import urls as taggit_templatetags2_urls
|
||||
import debug_toolbar
|
||||
|
||||
|
||||
urlpatterns = [
|
||||
url(r'^$', views.post_list, name='post_list'),
|
||||
url(r'^$', views.tag_list, name='tag_list'),
|
||||
url(r'^tag/(?P<slug>[-\w]+)/$', views.post_list, name='post_list_by_tag'),
|
||||
url(r'^taglist/$', views.tag_list, name='tag_list'),
|
||||
url(r'^student/', views.student_page, name='student_page'),
|
||||
url(r'^search/', views.blog_search_list_view, name='blog_search_list_view'),
|
||||
url(r'^search/result/$', views.blog_search_list_view, name='post_list_by_search'),
|
||||
url(r'^posts/$', views.post_list, name='post_list'),
|
||||
url(r'^search/', views.search_add, name='search_add'),
|
||||
url(r'^search/result/$', views.search_add, name='post_list_by_search'),
|
||||
url(r'^post/(?P<pk>\d+)/$', views.post_detail, name='post_detail'),
|
||||
url(r'^post/new/$', views.post_new, name='post_new'),
|
||||
url(r'^post/(?P<pk>\d+)/edit/$', views.post_edit, name='post_edit'),
|
||||
url(r'^drafts/$', views.post_draft_list, name='post_draft_list'),
|
||||
url(r'^post/(?P<pk>\d+)/publish/$', views.post_publish, name='post_publish'),
|
||||
url(r'^post/(?P<pk>\d+)/remove/$', views.post_remove, name='post_remove'),
|
||||
url(r'^student/(?P<slug>[-\w]+)/remove/$', views.tag_remove, name='tag_remove'),
|
||||
url(r'^tags/', include('taggit_templatetags2.urls')),
|
||||
]
|
||||
|
||||
#external code
|
||||
#import from https://django-debug-toolbar.readthedocs.io/en/latest/installation.html
|
||||
if settings.DEBUG:
|
||||
import debug_toolbar
|
||||
urlpatterns += [
|
||||
|
@ -1,29 +1,24 @@
|
||||
from django.shortcuts import render, get_object_or_404
|
||||
from django.utils import timezone
|
||||
from .models import Post, CustomUser
|
||||
from taggit.models import Tag
|
||||
from django.core.paginator import Paginator, EmptyPage, PageNotAnInteger
|
||||
from .forms import PostForm, NewTagForm
|
||||
from django.shortcuts import redirect
|
||||
from django.contrib.auth.decorators import login_required
|
||||
from django.contrib.admin.views.decorators import staff_member_required
|
||||
from django.contrib.auth import authenticate, login, logout
|
||||
from django.db.models import Q
|
||||
import sys
|
||||
import collections
|
||||
from taggit_templatetags2.views import TagCanvasListView
|
||||
from django.contrib.auth.models import User
|
||||
from django.contrib import messages
|
||||
|
||||
|
||||
|
||||
import logging
|
||||
import mysite.settings
|
||||
import operator
|
||||
|
||||
from django.shortcuts import render, get_object_or_404
|
||||
from django.contrib.auth import authenticate, login, logout
|
||||
from django.contrib.auth.decorators import login_required
|
||||
from django.shortcuts import redirect
|
||||
from django.utils import timezone
|
||||
from django.contrib.admin.views.decorators import staff_member_required
|
||||
from django.contrib.auth.models import User
|
||||
|
||||
from .models import Post, CustomUser
|
||||
from taggit.models import Tag
|
||||
from .forms import PostForm, NewTagForm
|
||||
from django.contrib import messages
|
||||
from post_office import mail
|
||||
from hitcount.models import HitCount
|
||||
from hitcount.views import HitCountMixin
|
||||
|
||||
|
||||
def navlogin(request):
|
||||
|
||||
log = logging.getLogger('mysite')
|
||||
logout(request)
|
||||
error = ""
|
||||
@ -45,22 +40,25 @@ def navlogin(request):
|
||||
context = {'error': error}
|
||||
return render(request, 'index.html', context)
|
||||
|
||||
|
||||
#external code customised
|
||||
#import from https://tutorial.djangogirls.org/en/django_forms/
|
||||
#start ---
|
||||
@login_required
|
||||
def post_list(request, slug=None):
|
||||
log = logging.getLogger('mysite')
|
||||
posts = Post.objects.filter(published_date__lte=timezone.now()).order_by('published_date')
|
||||
posts = Post.objects.filter(published_date__lte=timezone.now()).order_by('-published_date')
|
||||
if slug:
|
||||
tag = get_object_or_404(Tag, slug=slug)
|
||||
posts = posts.filter(tags__in=[tag])
|
||||
log.info(posts)
|
||||
return render(request, 'post_list.html', locals())
|
||||
|
||||
|
||||
@login_required
|
||||
def post_detail(request, pk):
|
||||
post = get_object_or_404(Post, pk=pk)
|
||||
return render(request, 'post_detail.html', {'post': post})
|
||||
hit_count = HitCount.objects.get_for_object(post)
|
||||
hit_count_response = HitCountMixin.hit_count(request, hit_count)
|
||||
return render(request, 'post_detail.html', locals())
|
||||
|
||||
|
||||
@login_required
|
||||
@ -72,6 +70,7 @@ def post_new(request):
|
||||
post = form.save(commit=False)
|
||||
post.author = request.user
|
||||
post.save()
|
||||
form.save_m2m()
|
||||
return redirect('post_detail', pk=post.pk)
|
||||
else:
|
||||
form = PostForm()
|
||||
@ -115,80 +114,62 @@ def post_publish(request, pk):
|
||||
@staff_member_required
|
||||
def post_remove(request, pk):
|
||||
post = get_object_or_404(Post, pk=pk)
|
||||
messages.info(request, 'Der Artikel "' + post.title + '" wurde gelöscht')
|
||||
post.delete()
|
||||
return redirect('post_list')
|
||||
|
||||
#end ---
|
||||
|
||||
@login_required
|
||||
def tag_remove(request, slug=None):
|
||||
if slug:
|
||||
tag = get_object_or_404(Tag, slug=slug)
|
||||
tag.delete()
|
||||
save_m2m()
|
||||
return redirect('student_page')
|
||||
def tag_list(request):
|
||||
log = logging.getLogger('mysite')
|
||||
u = User.objects.get(username=request.user)
|
||||
posts = Post.objects.none()
|
||||
if u:
|
||||
tags_user = Tag.objects.filter(customuser__user = u)
|
||||
log.info(tags_user)
|
||||
for tag in tags_user:
|
||||
query = Post.objects.filter(tags__in=[tag]).order_by('-published_date')
|
||||
posts = list(set(posts)|set(query))
|
||||
log.info(posts)
|
||||
return render(request, 'tag_list.html', {'posts': posts})
|
||||
|
||||
@login_required
|
||||
def student_page(request):
|
||||
def search_add(request):
|
||||
posts = Post.objects.filter(published_date__lte=timezone.now()).order_by('published_date')
|
||||
if request.method == 'GET':
|
||||
search_query = request.GET.get('search_box', None)
|
||||
posts = posts.filter(tags__name__in=[search_query])
|
||||
u = User.objects.get(username=request.user)
|
||||
if u:
|
||||
tagsuser = Tag.objects.filter(customuser__user = u)
|
||||
user_instance = get_object_or_404(CustomUser, user=request.user)
|
||||
if request.method == "POST":
|
||||
form = NewTagForm(request.POST, instance=user_instance)
|
||||
if form.is_valid():
|
||||
obj = form.save(commit=False)
|
||||
obj.user = request.user
|
||||
tag_names = [tag.name for tag in Tag.objects.all()]
|
||||
m_tags = form.cleaned_data['tags']
|
||||
m_tags = ' '.join(str(m_tags) for m_tags in m_tags)
|
||||
if m_tags in tag_names:
|
||||
obj.tags.add(m_tags)
|
||||
obj.save()
|
||||
m_tags = form.cleaned_data['m_tags']
|
||||
tags = Tag.objects.all()
|
||||
if m_tags in tags:
|
||||
obj.tags.add(*m_tags)
|
||||
form.save_m2m()
|
||||
return redirect('/student/')
|
||||
messages.info(request, 'Der Tag "' + m_tags + '" wurde gespeichert')
|
||||
return redirect('/search/')
|
||||
else:
|
||||
messages.info(request, 'Sorry !! Den Tag den du suchst gibt es leider nicht!')
|
||||
else:
|
||||
form = NewTagForm()
|
||||
return render(request, 'student_page.html', {'form':form})
|
||||
return render(request, 'search_add.html', locals())
|
||||
|
||||
@login_required
|
||||
def tag_list(request):
|
||||
log = logging.getLogger('mysite')
|
||||
u = User.objects.get(username=request.user)
|
||||
if u:
|
||||
tags_user = Tag.objects.filter(customuser__user = u)
|
||||
log.info(tags_user)
|
||||
for tag in tags_user:
|
||||
posts = Post.objects.filter(tags__in=[tag]).order_by('-published_date')
|
||||
log.info(posts)
|
||||
return render(request, 'tag_list.html', locals())
|
||||
|
||||
class TagSearch(TagCanvasListView):
|
||||
template_name = 'blog_search_list_view.html'
|
||||
model = Post
|
||||
context_object_name = 'list'
|
||||
paginate_by = 8
|
||||
|
||||
def get_context_data(self, **kwargs):
|
||||
context = super(CustomerSearch, self).get_context_data(**kwargs)
|
||||
context = self.get_queryset()
|
||||
return context
|
||||
|
||||
def get_queryset(self):
|
||||
cObj = Customer.objects.all()
|
||||
var_get_search = self.request.GET.get('search_box')
|
||||
|
||||
if var_get_search is not None:
|
||||
cObj = cObj.filter(firstname__icontains=var_get_search)
|
||||
if var_get_order_by is not None:
|
||||
cObj = cObj.order_by(var_get_order_by)
|
||||
return cObj
|
||||
|
||||
def blog_search_list_view(request):
|
||||
log = logging.getLogger('mysite')
|
||||
posts = Post.objects.filter(published_date__lte=timezone.now()).order_by('published_date')
|
||||
if request.method == 'GET':
|
||||
search_query = request.GET.get('search_box', None)
|
||||
log.info(search_query)
|
||||
posts = posts.filter(tags__name__in=[search_query])
|
||||
return render(request, 'blog_search_list_view.html', locals())
|
||||
|
||||
def tag_cloud(request):
|
||||
return render(request, 'tag_cloud.html', {})
|
||||
#external code customised
|
||||
#import from https://github.com/ui/django-post_office
|
||||
mail.send(
|
||||
# List of email addresses also accepted
|
||||
'kleinhenz.e@gmail.com',
|
||||
'esther.kleinhenz@web.de',
|
||||
subject='My email',
|
||||
message='Hi there!',
|
||||
html_message='Hi <strong>there</strong>!',
|
||||
)
|
||||
|
1
croniter
@ -1 +0,0 @@
|
||||
Subproject commit 3273cbc777423138f9b6cfb127de503b63ccd22f
|
BIN
doc/BA-Verteidigung.key
Executable file
BIN
doc/BA-Verteidigung.pdf
Normal file
BIN
doc/BA-Verteidigung.pptx
Normal file
BIN
doc/Verteidigung-Notizen.pages
Normal file
BIN
doc/Verteidigung-Notizen.pdf
Normal file
BIN
doc/abonnieren_video.mov
Normal file
BIN
doc/abonnieren_video_slow.mov
Normal file
@ -0,0 +1,36 @@
|
||||
\relax
|
||||
\providecommand\hyper@newdestlabel[2]{}
|
||||
\@writefile{toc}{\contentsline {chapter}{Acknowledgements}{iii}{chapter*.2}}
|
||||
\@setckpt{AcknowledgmentsDedicationSentence/acknowledgements}{
|
||||
\setcounter{page}{4}
|
||||
\setcounter{equation}{0}
|
||||
\setcounter{enumi}{0}
|
||||
\setcounter{enumii}{0}
|
||||
\setcounter{enumiii}{0}
|
||||
\setcounter{enumiv}{0}
|
||||
\setcounter{footnote}{0}
|
||||
\setcounter{mpfootnote}{0}
|
||||
\setcounter{part}{0}
|
||||
\setcounter{chapter}{0}
|
||||
\setcounter{section}{0}
|
||||
\setcounter{subsection}{0}
|
||||
\setcounter{subsubsection}{0}
|
||||
\setcounter{paragraph}{0}
|
||||
\setcounter{subparagraph}{0}
|
||||
\setcounter{figure}{0}
|
||||
\setcounter{table}{0}
|
||||
\setcounter{float@type}{8}
|
||||
\setcounter{parentequation}{0}
|
||||
\setcounter{lstnumber}{1}
|
||||
\setcounter{ContinuedFloat}{0}
|
||||
\setcounter{subfigure}{0}
|
||||
\setcounter{subtable}{0}
|
||||
\setcounter{r@tfl@t}{0}
|
||||
\setcounter{Item}{0}
|
||||
\setcounter{Hfootnote}{0}
|
||||
\setcounter{Hy@AnnotLevel}{0}
|
||||
\setcounter{bookmark@seq@number}{2}
|
||||
\setcounter{NAT@ctr}{0}
|
||||
\setcounter{lstlisting}{0}
|
||||
\setcounter{section@level}{0}
|
||||
}
|
@ -1,8 +1,8 @@
|
||||
\relax
|
||||
\providecommand\hyper@newdestlabel[2]{}
|
||||
\@writefile{toc}{\contentsline {chapter}{Abstract}{i}{figure.caption.1}}
|
||||
\@writefile{toc}{\contentsline {chapter}{Abstract}{ii}{Doc-Start}}
|
||||
\@setckpt{abstract/abstract}{
|
||||
\setcounter{page}{2}
|
||||
\setcounter{page}{3}
|
||||
\setcounter{equation}{0}
|
||||
\setcounter{enumi}{0}
|
||||
\setcounter{enumii}{0}
|
||||
@ -20,17 +20,18 @@
|
||||
\setcounter{figure}{0}
|
||||
\setcounter{table}{0}
|
||||
\setcounter{float@type}{8}
|
||||
\setcounter{Item}{0}
|
||||
\setcounter{Hfootnote}{0}
|
||||
\setcounter{Hy@AnnotLevel}{0}
|
||||
\setcounter{bookmark@seq@number}{1}
|
||||
\setcounter{parentequation}{0}
|
||||
\setcounter{lstnumber}{1}
|
||||
\setcounter{ContinuedFloat}{0}
|
||||
\setcounter{subfigure}{0}
|
||||
\setcounter{subtable}{0}
|
||||
\setcounter{r@tfl@t}{0}
|
||||
\setcounter{Item}{0}
|
||||
\setcounter{Hfootnote}{0}
|
||||
\setcounter{Hy@AnnotLevel}{0}
|
||||
\setcounter{bookmark@seq@number}{1}
|
||||
\setcounter{AM@survey}{0}
|
||||
\setcounter{NAT@ctr}{0}
|
||||
\setcounter{lstlisting}{0}
|
||||
\setcounter{section@level}{0}
|
||||
\setcounter{lstlisting}{0}
|
||||
}
|
||||
|
@ -2,7 +2,6 @@
|
||||
\providecommand\hyper@newdestlabel[2]{}
|
||||
\providecommand*\new@tpo@label[2]{}
|
||||
\catcode `"\active
|
||||
\AC@reset@newl@bel
|
||||
\providecommand\HyperFirstAtBeginDocument{\AtBeginDocument}
|
||||
\HyperFirstAtBeginDocument{\ifx\hyper@anchor\@undefined
|
||||
\global\let\oldcontentsline\contentsline
|
||||
@ -18,14 +17,15 @@
|
||||
\global\let\hyper@last\relax
|
||||
\gdef\HyperFirstAtBeginDocument#1{#1}
|
||||
\providecommand*\HyPL@Entry[1]{}
|
||||
\AC@reset@newl@bel
|
||||
\@input{titlepage/titlepage.aux}
|
||||
\HyPL@Entry{0<</S/D>>}
|
||||
\babel@aux{german}{}
|
||||
\babel@aux{german}{}
|
||||
\HyPL@Entry{1<</S/r>>}
|
||||
\@input{abstract/abstract.aux}
|
||||
\HyPL@Entry{2<</S/r>>}
|
||||
\HyPL@Entry{6<</S/D>>}
|
||||
\@writefile{toc}{\contentsline {chapter}{Abbildungsverzeichnis}{1}{chapter*.4}}
|
||||
\HyPL@Entry{5<</S/D>>}
|
||||
\@writefile{toc}{\contentsline {chapter}{Abbildungsverzeichnis}{1}{chapter*.3}}
|
||||
\@input{chapters/einleitung.aux}
|
||||
\@input{chapters/framework.aux}
|
||||
\@input{chapters/prototyp.aux}
|
||||
@ -35,13 +35,33 @@
|
||||
\bibstyle{alpha}
|
||||
\citation{*}
|
||||
\bibdata{references/References_2}
|
||||
\bibcite{Dixit}{{Dix18}{}{{}}{{}}}
|
||||
\bibcite{Python}{{Fou18}{}{{}}{{}}}
|
||||
\bibcite{Bootstrap}{{BA11}{}{{}}{{}}}
|
||||
\bibcite{Computer}{{Com18}{}{{}}{{}}}
|
||||
\bibcite{Djangogirls}{{Dja18a}{}{{}}{{}}}
|
||||
\bibcite{Djangogirls-2}{{Dja18b}{}{{}}{{}}}
|
||||
\bibcite{Moodle}{{Dok15}{}{{}}{{}}}
|
||||
\bibcite{Moodle-2}{{Dok18}{}{{}}{{}}}
|
||||
\bibcite{Fiok}{{Fio14}{}{{}}{{}}}
|
||||
\bibcite{Django-4}{{Fou18a}{}{{}}{{}}}
|
||||
\bibcite{Django}{{Fou18b}{}{{}}{{}}}
|
||||
\bibcite{Django-2}{{Fou18c}{}{{}}{{}}}
|
||||
\bibcite{Django-3}{{Fou18d}{}{{}}{{}}}
|
||||
\bibcite{Python}{{Fou18e}{}{{}}{{}}}
|
||||
\@writefile{toc}{\contentsline {chapter}{Referenzen}{33}{chapter*.24}}
|
||||
\bibcite{Gaynor}{{Gay10}{}{{}}{{}}}
|
||||
\bibcite{Sario}{{Gmb18}{}{{}}{{}}}
|
||||
\bibcite{Hellmuth}{{Hel12}{}{{}}{{}}}
|
||||
\bibcite{Herzog}{{Her16}{}{{}}{{}}}
|
||||
\bibcite{King}{{Kin17}{}{{}}{{}}}
|
||||
\bibcite{Leipner}{{Lei13}{}{{}}{{}}}
|
||||
\bibcite{Ndukwe}{{Ndu17}{}{{}}{{}}}
|
||||
\bibcite{Nevius}{{Nev15}{}{{}}{{}}}
|
||||
\bibcite{Ong}{{Ong18}{}{{}}{{}}}
|
||||
\bibcite{Otto}{{Ott11}{}{{}}{{}}}
|
||||
\bibcite{Otto-2}{{Ott12}{}{{}}{{}}}
|
||||
\bibcite{Shabda}{{Sha09}{}{{}}{{}}}
|
||||
\bibcite{Shelest}{{She09}{}{{}}{{}}}
|
||||
\bibcite{Solutions}{{Sol17}{}{{}}{{}}}
|
||||
\bibcite{Timm}{{Tim15}{}{{}}{{}}}
|
||||
\bibcite{Weigend}{{Wei17}{}{{}}{{}}}
|
||||
\providecommand\NAT@force@numbers{}\NAT@force@numbers
|
||||
\@writefile{toc}{\contentsline {chapter}{Referenzen}{19}{chapter*.11}}
|
||||
|
@ -1,13 +1,75 @@
|
||||
\begin{thebibliography}{Ndu17}
|
||||
\begin{thebibliography}{Fou18d}
|
||||
|
||||
\bibitem[Dix18]{Dixit}
|
||||
Chitrank Dixit.
|
||||
\newblock Pep-8 tutorial: Code standards in python.
|
||||
\bibitem[BA11]{Bootstrap}
|
||||
Twitter~Inc Bootstrap~Authors.
|
||||
\newblock Bootstrap repository.
|
||||
\newblock 2011.
|
||||
\newblock https://github.com/twbs/bootstrap.
|
||||
|
||||
\bibitem[Com18]{Computer}
|
||||
The Computer~Language Company.
|
||||
\newblock Definition of: user permissions.
|
||||
\newblock 2018.
|
||||
\newblock https://www.pcmag.com/encyclopedia/term/58231/user-permissions.
|
||||
|
||||
\bibitem[Dja18a]{Djangogirls}
|
||||
Djangogirls.
|
||||
\newblock Creating a blog post model.
|
||||
\newblock 2018.
|
||||
\newblock https://tutorial.djangogirls.org/en/django\_models/.
|
||||
|
||||
\bibitem[Dja18b]{Djangogirls-2}
|
||||
Djangogirls.
|
||||
\newblock We inspire women to fall in love with programming.
|
||||
\newblock 2018.
|
||||
\newblock https://djangogirls.org/.
|
||||
|
||||
\bibitem[Dok15]{Moodle}
|
||||
Moodle Dokumentation.
|
||||
\newblock Aufbau einer moodle-site.
|
||||
\newblock 2015.
|
||||
\newblock https://docs.moodle.org/35/de/Aufbau\_einer\_Moodle-Site.
|
||||
|
||||
\bibitem[Dok18]{Moodle-2}
|
||||
Moodle Dokumentation.
|
||||
\newblock Cron-job.
|
||||
\newblock 2018.
|
||||
\newblock https://docs.moodle.org/35/de/Cron-Job.
|
||||
|
||||
\bibitem[Fio14]{Fiok}
|
||||
Marzena Fiok.
|
||||
\newblock E-mail-flut sorgt für kostenlawine.
|
||||
\newblock 2014.
|
||||
\newblock
|
||||
https://www.tecchannel.de/a/e-mail-flut-sorgt-fuer-kostenlawine,402338,3.
|
||||
|
||||
\bibitem[Fou18a]{Django-4}
|
||||
Django~Software Foundation.
|
||||
\newblock Cross site request forgery protection.
|
||||
\newblock 2018.
|
||||
\newblock https://docs.djangoproject.com/en/dev/ref/csrf/.
|
||||
|
||||
\bibitem[Fou18b]{Django}
|
||||
Django~Software Foundation.
|
||||
\newblock django.contrib.auth, user model.
|
||||
\newblock 2018.
|
||||
\newblock https://docs.djangoproject.com/en/2.1/ref/contrib/auth/.
|
||||
|
||||
\bibitem[Fou18c]{Django-2}
|
||||
Django~Software Foundation.
|
||||
\newblock Modelforms - the save() methode.
|
||||
\newblock 2018.
|
||||
\newblock
|
||||
https://www.datacamp.com/community/tutorials/pep8-tutorial-python-code.
|
||||
https://docs.djangoproject.com/en/dev/topics/forms/modelforms/\#the-save-method.
|
||||
|
||||
\bibitem[Fou18]{Python}
|
||||
\bibitem[Fou18d]{Django-3}
|
||||
Django~Software Foundation.
|
||||
\newblock Working with forms.
|
||||
\newblock 2018.
|
||||
\newblock
|
||||
https://docs.djangoproject.com/en/dev/topics/forms/\#using-a-form-in-a-view.
|
||||
|
||||
\bibitem[Fou18e]{Python}
|
||||
Python~Software Foundation.
|
||||
\newblock Virtual environments and packages.
|
||||
\newblock 2018.
|
||||
@ -19,6 +81,19 @@ Alex Gaynor.
|
||||
\newblock 2010.
|
||||
\newblock https://django-taggit.readthedocs.io/en/latest/getting\_started.
|
||||
|
||||
\bibitem[Gmb18]{Sario}
|
||||
Sario~Marketing GmbH.
|
||||
\newblock Tagging.
|
||||
\newblock 2018.
|
||||
\newblock https://www.textbroker.de/tagging.
|
||||
|
||||
\bibitem[Hel12]{Hellmuth}
|
||||
Dirk Hellmuth.
|
||||
\newblock Wenn e-mails zur last werden: Wege aus der mail-Überflutung.
|
||||
\newblock 2012.
|
||||
\newblock
|
||||
https://www.unternehmer.de/management-people-skills/138949-wenn-e-mails-zur-last-werden-wege-aus-der-mail-ueberflutung.
|
||||
|
||||
\bibitem[Her16]{Herzog}
|
||||
Stephan Herzog.
|
||||
\newblock Model view controller, model view presenter, and model view viewmodel
|
||||
@ -50,6 +125,38 @@ Nnenna Ndukwe.
|
||||
\newblock
|
||||
https://medium.com/@nnennahacks/https-medium-com-nnennandukwe-python-is-the-back-end-programming-language-of-the-future-heres-why.
|
||||
|
||||
\bibitem[Nev15]{Nevius}
|
||||
Ryan Nevius.
|
||||
\newblock django-post\_office git repository.
|
||||
\newblock 2015.
|
||||
\newblock https://ryannevius.com.
|
||||
|
||||
\bibitem[Ong18]{Ong}
|
||||
Selwin Ong.
|
||||
\newblock Django request-response cycle.
|
||||
\newblock 2018.
|
||||
\newblock https://github.com/ui/django-post\_office/blob/master/AUTHORS.rst.
|
||||
|
||||
\bibitem[Ott11]{Otto}
|
||||
Mark Otto.
|
||||
\newblock Bootstrap from twitter.
|
||||
\newblock 2011.
|
||||
\newblock
|
||||
https://blog.twitter.com/developer/en\_us/a/2011/bootstrap-twitter.html.
|
||||
|
||||
\bibitem[Ott12]{Otto-2}
|
||||
Mark Otto.
|
||||
\newblock Say hello to bootstrap 2.0.
|
||||
\newblock 2012.
|
||||
\newblock
|
||||
https://web.archive.org/web/20120203191214/https://dev.twitter.com/blog/say-hello-to-bootstrap-2.
|
||||
|
||||
\bibitem[Sha09]{Shabda}
|
||||
Shabda.
|
||||
\newblock Understanding decorators.
|
||||
\newblock 2009.
|
||||
\newblock https://www.agiliq.com/blog/2009/06/understanding-decorators/.
|
||||
|
||||
\bibitem[She09]{Shelest}
|
||||
Alexy Shelest.
|
||||
\newblock Model view controller, model view presenter, and model view viewmodel
|
||||
@ -58,4 +165,23 @@ Alexy Shelest.
|
||||
\newblock
|
||||
https://www.codeproject.com/Articles/42830/Model-View-Controller-Model-View-Presenter-and-Mod.
|
||||
|
||||
\bibitem[Sol17]{Solutions}
|
||||
Mindfire Solutions.
|
||||
\newblock Advantages and disadvantages of python programming language.
|
||||
\newblock 2017.
|
||||
\newblock
|
||||
https://medium.com/@mindfiresolutions.usa/advantages-and-disadvantages-of-python-programming-language-fd0b394f2121.
|
||||
|
||||
\bibitem[Tim15]{Timm}
|
||||
Damon Timm.
|
||||
\newblock django-hitcount documentation.
|
||||
\newblock 2015.
|
||||
\newblock https://django-hitcount.readthedocs.io/en/latest/overview.html.
|
||||
|
||||
\bibitem[Wei17]{Weigend}
|
||||
Michael Weigend.
|
||||
\newblock {\em Python GE-PACKT}.
|
||||
\newblock 2017.
|
||||
\newblock Kapitel 23.1.
|
||||
|
||||
\end{thebibliography}
|
||||
|
@ -11,53 +11,77 @@ A level-1 auxiliary file: chapters/ausblick.aux
|
||||
A level-1 auxiliary file: chapters/fazit.aux
|
||||
The style file: alpha.bst
|
||||
Database file #1: ../references/References_2.bib
|
||||
Warning--empty journal in Dixit
|
||||
Repeated entry---line 176 of file ../references/References_2.bib
|
||||
: @article{Djangogirls-2
|
||||
: ,
|
||||
I'm skipping whatever remains of this entry
|
||||
Warning--empty journal in Bootstrap
|
||||
Warning--empty journal in Computer
|
||||
Warning--empty journal in Djangogirls
|
||||
Warning--empty journal in Djangogirls-2
|
||||
Warning--empty journal in Moodle
|
||||
Warning--empty journal in Moodle-2
|
||||
Warning--empty journal in Fiok
|
||||
Warning--empty journal in Django-4
|
||||
Warning--empty journal in Django
|
||||
Warning--empty journal in Django-2
|
||||
Warning--empty journal in Django-3
|
||||
Warning--empty journal in Python
|
||||
Warning--empty journal in Gaynor
|
||||
Warning--empty journal in Sario
|
||||
Warning--empty journal in Hellmuth
|
||||
Warning--empty journal in Herzog
|
||||
Warning--empty journal in King
|
||||
Warning--empty journal in Leipner
|
||||
Warning--empty journal in Ndukwe
|
||||
Warning--empty journal in Nevius
|
||||
Warning--empty journal in Ong
|
||||
Warning--empty journal in Otto
|
||||
Warning--empty journal in Otto-2
|
||||
Warning--empty journal in Shabda
|
||||
Warning--empty journal in Shelest
|
||||
You've used 8 entries,
|
||||
Warning--empty journal in Solutions
|
||||
Warning--empty journal in Timm
|
||||
Warning--empty publisher in Weigend
|
||||
You've used 28 entries,
|
||||
2543 wiz_defined-function locations,
|
||||
611 strings with 6085 characters,
|
||||
and the built_in function-call counts, 2196 in all, are:
|
||||
= -- 216
|
||||
> -- 64
|
||||
< -- 8
|
||||
+ -- 16
|
||||
- -- 16
|
||||
* -- 106
|
||||
:= -- 390
|
||||
add.period$ -- 32
|
||||
call.type$ -- 8
|
||||
change.case$ -- 40
|
||||
chr.to.int$ -- 8
|
||||
cite$ -- 16
|
||||
duplicate$ -- 112
|
||||
empty$ -- 161
|
||||
format.name$ -- 32
|
||||
if$ -- 426
|
||||
int.to.chr$ -- 1
|
||||
707 strings with 8346 characters,
|
||||
and the built_in function-call counts, 7629 in all, are:
|
||||
= -- 753
|
||||
> -- 224
|
||||
< -- 28
|
||||
+ -- 61
|
||||
- -- 56
|
||||
* -- 368
|
||||
:= -- 1334
|
||||
add.period$ -- 112
|
||||
call.type$ -- 28
|
||||
change.case$ -- 139
|
||||
chr.to.int$ -- 23
|
||||
cite$ -- 56
|
||||
duplicate$ -- 394
|
||||
empty$ -- 566
|
||||
format.name$ -- 111
|
||||
if$ -- 1489
|
||||
int.to.chr$ -- 6
|
||||
int.to.str$ -- 0
|
||||
missing$ -- 8
|
||||
newline$ -- 51
|
||||
num.names$ -- 24
|
||||
pop$ -- 48
|
||||
missing$ -- 29
|
||||
newline$ -- 171
|
||||
num.names$ -- 84
|
||||
pop$ -- 170
|
||||
preamble$ -- 1
|
||||
purify$ -- 48
|
||||
purify$ -- 168
|
||||
quote$ -- 0
|
||||
skip$ -- 88
|
||||
skip$ -- 302
|
||||
stack$ -- 0
|
||||
substring$ -- 56
|
||||
swap$ -- 0
|
||||
text.length$ -- 8
|
||||
text.prefix$ -- 8
|
||||
substring$ -- 196
|
||||
swap$ -- 1
|
||||
text.length$ -- 28
|
||||
text.prefix$ -- 27
|
||||
top$ -- 0
|
||||
type$ -- 64
|
||||
warning$ -- 8
|
||||
while$ -- 16
|
||||
width$ -- 10
|
||||
write$ -- 106
|
||||
(There were 8 warnings)
|
||||
type$ -- 220
|
||||
warning$ -- 28
|
||||
while$ -- 56
|
||||
width$ -- 34
|
||||
write$ -- 366
|
||||
(There was 1 error message)
|
||||
|
@ -2,12 +2,26 @@
|
||||
\babel@toc {german}{}
|
||||
\addvspace {10\p@ }
|
||||
\addvspace {10\p@ }
|
||||
\contentsline {figure}{\numberline {2.1}{\ignorespaces Vereinfachter MVP\relax }}{6}{figure.caption.5}
|
||||
\contentsline {figure}{\numberline {2.2}{\ignorespaces Request-Response-Kreislauf des Django Frameworks\relax }}{7}{figure.caption.6}
|
||||
\contentsline {figure}{\numberline {2.3}{\ignorespaces Erstellen der virtuelle Umgebung im Terminal\relax }}{8}{figure.caption.7}
|
||||
\contentsline {figure}{\numberline {2.4}{\ignorespaces Beispiel eines LDAP-Trees\relax }}{9}{figure.caption.8}
|
||||
\contentsline {figure}{\numberline {2.5}{\ignorespaces Einbindung von Bootstrap in einer HTML-Datei\relax }}{11}{figure.caption.9}
|
||||
\contentsline {figure}{\numberline {2.6}{\ignorespaces Bootstrap-Klassen in HTML-Tag\relax }}{12}{figure.caption.10}
|
||||
\contentsline {figure}{\numberline {2.1}{\ignorespaces Vereinfachter MVP [She09]\relax }}{5}{figure.caption.4}
|
||||
\contentsline {figure}{\numberline {2.2}{\ignorespaces Request-Response-Kreislauf des Django Frameworks [Nev15]\relax }}{6}{figure.caption.5}
|
||||
\contentsline {figure}{\numberline {2.3}{\ignorespaces Erstellen der virtuelle Umgebung im Terminal\relax }}{7}{figure.caption.6}
|
||||
\contentsline {figure}{\numberline {2.4}{\ignorespaces Einbindung von Bootstrap in einer HTML-Datei\relax }}{11}{figure.caption.7}
|
||||
\contentsline {figure}{\numberline {2.5}{\ignorespaces Bootstrap-Klassen in HTML-Tag\relax }}{11}{figure.caption.8}
|
||||
\addvspace {10\p@ }
|
||||
\contentsline {figure}{\numberline {3.1}{\ignorespaces Forschungsdesign\relax }}{14}{figure.caption.9}
|
||||
\contentsline {figure}{\numberline {3.2}{\ignorespaces CustomUserModel in models.py\relax }}{16}{figure.caption.10}
|
||||
\contentsline {figure}{\numberline {3.3}{\ignorespaces Datenmodellierung von \texttt {User} und \texttt {Post}\relax }}{17}{figure.caption.11}
|
||||
\contentsline {figure}{\numberline {3.4}{\ignorespaces Menü für eingeloggte Benutzer mit Adminrechte\relax }}{18}{figure.caption.12}
|
||||
\contentsline {figure}{\numberline {3.5}{\ignorespaces User Stories\relax }}{19}{figure.caption.13}
|
||||
\contentsline {figure}{\numberline {3.6}{\ignorespaces Prototyp Artikel-Editor.\relax }}{21}{figure.caption.14}
|
||||
\contentsline {figure}{\numberline {3.7}{\ignorespaces Funktion post\_edit, Auszug aus views.py.\relax }}{21}{figure.caption.15}
|
||||
\contentsline {figure}{\numberline {3.8}{\ignorespaces Prototyp Suche- und Abonnier-Seite\relax }}{22}{figure.caption.16}
|
||||
\contentsline {figure}{\numberline {3.9}{\ignorespaces Funktion search\_add, Auszug aus views.py.\relax }}{23}{figure.caption.17}
|
||||
\contentsline {figure}{\numberline {3.10}{\ignorespaces Prototyp Newsfeed Seite\relax }}{24}{figure.caption.18}
|
||||
\addvspace {10\p@ }
|
||||
\contentsline {figure}{\numberline {4.1}{\ignorespaces Vergleich der hochschulinternen E-Mails\relax }}{26}{figure.caption.19}
|
||||
\contentsline {figure}{\numberline {4.2}{\ignorespaces Details der relevanten E-Mails des Probanden\relax }}{27}{figure.caption.20}
|
||||
\contentsline {figure}{\numberline {4.3}{\ignorespaces Details der irrelevanten E-Mails des Probanden\relax }}{28}{figure.caption.21}
|
||||
\contentsline {figure}{\numberline {4.4}{\ignorespaces Vergleich der Anzahl von eintreffenden E-Mails zwischen aktueller Situation und unter Verwendung des Prototyps\relax }}{28}{figure.caption.22}
|
||||
\addvspace {10\p@ }
|
||||
\contentsline {figure}{\numberline {5.1}{\ignorespaces Cron-Tab der im Prototyp getesteten Benachrichtigung.\relax }}{32}{figure.caption.23}
|
||||
|
@ -1,23 +1,29 @@
|
||||
\BOOKMARK [0][-]{figure.caption.1}{Abstract}{}% 1
|
||||
\BOOKMARK [0][-]{chapter*.4}{Abbildungsverzeichnis}{}% 2
|
||||
\BOOKMARK [0][-]{Doc-Start}{Abstract}{}% 1
|
||||
\BOOKMARK [0][-]{chapter*.3}{Abbildungsverzeichnis}{}% 2
|
||||
\BOOKMARK [0][-]{chapter.1}{Einleitung}{}% 3
|
||||
\BOOKMARK [1][-]{section.1.1}{Ausgangssituation}{chapter.1}% 4
|
||||
\BOOKMARK [1][-]{section.1.2}{Ziel\040der\040Arbeit}{chapter.1}% 5
|
||||
\BOOKMARK [0][-]{chapter.2}{Framework}{}% 6
|
||||
\BOOKMARK [1][-]{section.2.1}{Django}{chapter.2}% 7
|
||||
\BOOKMARK [2][-]{subsection.2.1.1}{Besonderheiten}{section.2.1}% 8
|
||||
\BOOKMARK [1][-]{section.2.2}{Erweiterungen}{chapter.2}% 9
|
||||
\BOOKMARK [2][-]{subsection.2.2.1}{Taggable-Manager}{section.2.2}% 10
|
||||
\BOOKMARK [1][-]{section.2.3}{Bootstrap}{chapter.2}% 11
|
||||
\BOOKMARK [0][-]{chapter.3}{Prototyp}{}% 12
|
||||
\BOOKMARK [1][-]{section.3.1}{Organisation}{chapter.3}% 13
|
||||
\BOOKMARK [2][-]{subsection.3.1.1}{Verwaltung\040im\040Administrator-Backend}{section.3.1}% 14
|
||||
\BOOKMARK [2][-]{subsection.3.1.2}{Berechtigung\040der\040User}{section.3.1}% 15
|
||||
\BOOKMARK [1][-]{section.3.2}{Funktion}{chapter.3}% 16
|
||||
\BOOKMARK [2][-]{subsection.3.2.1}{Abonnieren}{section.3.2}% 17
|
||||
\BOOKMARK [2][-]{subsection.3.2.2}{Filtern}{section.3.2}% 18
|
||||
\BOOKMARK [2][-]{subsection.3.2.3}{Benachrichtigung}{section.3.2}% 19
|
||||
\BOOKMARK [0][-]{chapter.4}{Ergebnis}{}% 20
|
||||
\BOOKMARK [1][-]{subsection.4.0.1}{Evaluierung}{chapter.4}% 21
|
||||
\BOOKMARK [0][-]{chapter.5}{Zusammenfassung\040und\040Ausblick}{}% 22
|
||||
\BOOKMARK [0][-]{chapter*.11}{Referenzen}{}% 23
|
||||
\BOOKMARK [2][-]{subsection.2.1.1}{Besonderheiten\040Djangos}{section.2.1}% 8
|
||||
\BOOKMARK [2][-]{subsection.2.1.2}{Virtuelle\040Umgebung}{section.2.1}% 9
|
||||
\BOOKMARK [2][-]{subsection.2.1.3}{Sicherheit}{section.2.1}% 10
|
||||
\BOOKMARK [1][-]{section.2.2}{Erweiterungen}{chapter.2}% 11
|
||||
\BOOKMARK [2][-]{subsection.2.2.1}{Taggable-Manager}{section.2.2}% 12
|
||||
\BOOKMARK [2][-]{subsection.2.2.2}{Hilfsbibliotheken}{section.2.2}% 13
|
||||
\BOOKMARK [1][-]{section.2.3}{Bootstrap}{chapter.2}% 14
|
||||
\BOOKMARK [0][-]{chapter.3}{Prototyp}{}% 15
|
||||
\BOOKMARK [1][-]{section.3.1}{Forschungsdesign}{chapter.3}% 16
|
||||
\BOOKMARK [1][-]{section.3.2}{Organisation}{chapter.3}% 17
|
||||
\BOOKMARK [2][-]{subsection.3.2.1}{Datenmodellierung}{section.3.2}% 18
|
||||
\BOOKMARK [2][-]{subsection.3.2.2}{Berechtigungen\040der\040User}{section.3.2}% 19
|
||||
\BOOKMARK [1][-]{section.3.3}{Funktionen}{chapter.3}% 20
|
||||
\BOOKMARK [2][-]{subsection.3.3.1}{Verwaltung\040der\040Funktionen}{section.3.3}% 21
|
||||
\BOOKMARK [2][-]{subsection.3.3.2}{Artikel\040abonnieren}{section.3.3}% 22
|
||||
\BOOKMARK [2][-]{subsection.3.3.3}{Filtern\040von\040Artikeln}{section.3.3}% 23
|
||||
\BOOKMARK [0][-]{chapter.4}{Evaluation}{}% 24
|
||||
\BOOKMARK [1][-]{subsection.4.0.1}{Ergebnis}{chapter.4}% 25
|
||||
\BOOKMARK [2][-]{subsection.4.0.2}{Diskussion}{subsection.4.0.1}% 26
|
||||
\BOOKMARK [0][-]{chapter.5}{Schlussbetrachtung}{}% 27
|
||||
\BOOKMARK [1][-]{subsection.5.0.1}{Ausblick}{chapter.5}% 28
|
||||
\BOOKMARK [0][-]{chapter*.24}{Referenzen}{}% 29
|
||||
|
@ -1,25 +1,31 @@
|
||||
\babel@toc {german}{}
|
||||
\babel@toc {german}{}
|
||||
\contentsline {chapter}{Abstract}{i}{figure.caption.1}
|
||||
\contentsline {chapter}{Abbildungsverzeichnis}{1}{chapter*.4}
|
||||
\contentsline {chapter}{\numberline {1}Einleitung}{3}{chapter.1}
|
||||
\contentsline {section}{\numberline {1.1}Ausgangssituation}{3}{section.1.1}
|
||||
\contentsline {section}{\numberline {1.2}Ziel der Arbeit}{4}{section.1.2}
|
||||
\contentsline {chapter}{\numberline {2}Framework}{5}{chapter.2}
|
||||
\contentsline {section}{\numberline {2.1}Django}{5}{section.2.1}
|
||||
\contentsline {subsection}{\numberline {2.1.1}Besonderheiten}{6}{subsection.2.1.1}
|
||||
\contentsline {chapter}{Abstract}{ii}{Doc-Start}
|
||||
\contentsline {chapter}{Abbildungsverzeichnis}{1}{chapter*.3}
|
||||
\contentsline {chapter}{\numberline {1}Einleitung}{2}{chapter.1}
|
||||
\contentsline {section}{\numberline {1.1}Ausgangssituation}{2}{section.1.1}
|
||||
\contentsline {section}{\numberline {1.2}Ziel der Arbeit}{3}{section.1.2}
|
||||
\contentsline {chapter}{\numberline {2}Framework}{4}{chapter.2}
|
||||
\contentsline {section}{\numberline {2.1}Django}{4}{section.2.1}
|
||||
\contentsline {subsection}{\numberline {2.1.1}Besonderheiten Djangos}{6}{subsection.2.1.1}
|
||||
\contentsline {subsection}{\numberline {2.1.2}Virtuelle Umgebung}{7}{subsection.2.1.2}
|
||||
\contentsline {subsection}{\numberline {2.1.3}Sicherheit}{8}{subsection.2.1.3}
|
||||
\contentsline {section}{\numberline {2.2}Erweiterungen}{8}{section.2.2}
|
||||
\contentsline {subsection}{\numberline {2.2.1}Taggable-Manager}{10}{subsection.2.2.1}
|
||||
\contentsline {section}{\numberline {2.3}Bootstrap}{11}{section.2.3}
|
||||
\contentsline {subsection}{\numberline {2.2.1}Taggable-Manager}{9}{subsection.2.2.1}
|
||||
\contentsline {subsection}{\numberline {2.2.2}Hilfsbibliotheken}{10}{subsection.2.2.2}
|
||||
\contentsline {section}{\numberline {2.3}Bootstrap}{10}{section.2.3}
|
||||
\contentsline {chapter}{\numberline {3}Prototyp}{13}{chapter.3}
|
||||
\contentsline {section}{\numberline {3.1}Organisation}{13}{section.3.1}
|
||||
\contentsline {subsection}{\numberline {3.1.1}Verwaltung im Administrator-Backend}{13}{subsection.3.1.1}
|
||||
\contentsline {subsection}{\numberline {3.1.2}Berechtigung der User}{13}{subsection.3.1.2}
|
||||
\contentsline {section}{\numberline {3.2}Funktion}{13}{section.3.2}
|
||||
\contentsline {subsection}{\numberline {3.2.1}Abonnieren}{13}{subsection.3.2.1}
|
||||
\contentsline {subsection}{\numberline {3.2.2}Filtern}{13}{subsection.3.2.2}
|
||||
\contentsline {subsection}{\numberline {3.2.3}Benachrichtigung}{14}{subsection.3.2.3}
|
||||
\contentsline {chapter}{\numberline {4}Ergebnis}{15}{chapter.4}
|
||||
\contentsline {subsection}{\numberline {4.0.1}Evaluierung}{15}{subsection.4.0.1}
|
||||
\contentsline {chapter}{\numberline {5}Zusammenfassung und Ausblick}{17}{chapter.5}
|
||||
\contentsline {chapter}{Referenzen}{19}{chapter*.11}
|
||||
\contentsline {section}{\numberline {3.1}Forschungsdesign}{13}{section.3.1}
|
||||
\contentsline {section}{\numberline {3.2}Organisation}{14}{section.3.2}
|
||||
\contentsline {subsection}{\numberline {3.2.1}Datenmodellierung}{15}{subsection.3.2.1}
|
||||
\contentsline {subsection}{\numberline {3.2.2}Berechtigungen der User}{16}{subsection.3.2.2}
|
||||
\contentsline {section}{\numberline {3.3}Funktionen}{18}{section.3.3}
|
||||
\contentsline {subsection}{\numberline {3.3.1}Verwaltung der Funktionen}{18}{subsection.3.3.1}
|
||||
\contentsline {subsection}{\numberline {3.3.2}Artikel abonnieren}{20}{subsection.3.3.2}
|
||||
\contentsline {subsection}{\numberline {3.3.3}Filtern von Artikeln}{23}{subsection.3.3.3}
|
||||
\contentsline {chapter}{\numberline {4}Evaluation}{26}{chapter.4}
|
||||
\contentsline {subsection}{\numberline {4.0.1}Ergebnis}{29}{subsection.4.0.1}
|
||||
\contentsline {subsection}{\numberline {4.0.2}Diskussion}{29}{subsection.4.0.2}
|
||||
\contentsline {chapter}{\numberline {5}Schlussbetrachtung}{31}{chapter.5}
|
||||
\contentsline {subsection}{\numberline {5.0.1}Ausblick}{31}{subsection.5.0.1}
|
||||
\contentsline {chapter}{Referenzen}{33}{chapter*.24}
|
||||
|
@ -1,7 +1,7 @@
|
||||
\relax
|
||||
\providecommand\hyper@newdestlabel[2]{}
|
||||
\@setckpt{chapters/ausblick}{
|
||||
\setcounter{page}{16}
|
||||
\setcounter{page}{31}
|
||||
\setcounter{equation}{0}
|
||||
\setcounter{enumi}{0}
|
||||
\setcounter{enumii}{0}
|
||||
@ -12,24 +12,25 @@
|
||||
\setcounter{part}{0}
|
||||
\setcounter{chapter}{4}
|
||||
\setcounter{section}{0}
|
||||
\setcounter{subsection}{1}
|
||||
\setcounter{subsection}{2}
|
||||
\setcounter{subsubsection}{0}
|
||||
\setcounter{paragraph}{0}
|
||||
\setcounter{subparagraph}{0}
|
||||
\setcounter{figure}{0}
|
||||
\setcounter{figure}{4}
|
||||
\setcounter{table}{0}
|
||||
\setcounter{float@type}{8}
|
||||
\setcounter{Item}{0}
|
||||
\setcounter{Hfootnote}{16}
|
||||
\setcounter{Hy@AnnotLevel}{0}
|
||||
\setcounter{bookmark@seq@number}{26}
|
||||
\setcounter{parentequation}{0}
|
||||
\setcounter{lstnumber}{1}
|
||||
\setcounter{ContinuedFloat}{0}
|
||||
\setcounter{subfigure}{0}
|
||||
\setcounter{subtable}{0}
|
||||
\setcounter{r@tfl@t}{0}
|
||||
\setcounter{Item}{0}
|
||||
\setcounter{Hfootnote}{0}
|
||||
\setcounter{Hy@AnnotLevel}{0}
|
||||
\setcounter{bookmark@seq@number}{21}
|
||||
\setcounter{AM@survey}{0}
|
||||
\setcounter{NAT@ctr}{0}
|
||||
\setcounter{lstlisting}{0}
|
||||
\setcounter{section@level}{0}
|
||||
\setcounter{lstlisting}{0}
|
||||
}
|
||||
|
@ -1,12 +1,12 @@
|
||||
\relax
|
||||
\providecommand\hyper@newdestlabel[2]{}
|
||||
\@writefile{toc}{\contentsline {chapter}{\numberline {1}Einleitung}{3}{chapter.1}}
|
||||
\@writefile{toc}{\contentsline {chapter}{\numberline {1}Einleitung}{2}{chapter.1}}
|
||||
\@writefile{lof}{\addvspace {10\p@ }}
|
||||
\@writefile{lot}{\addvspace {10\p@ }}
|
||||
\@writefile{toc}{\contentsline {section}{\numberline {1.1}Ausgangssituation}{3}{section.1.1}}
|
||||
\@writefile{toc}{\contentsline {section}{\numberline {1.2}Ziel der Arbeit}{4}{section.1.2}}
|
||||
\@writefile{toc}{\contentsline {section}{\numberline {1.1}Ausgangssituation}{2}{section.1.1}}
|
||||
\@writefile{toc}{\contentsline {section}{\numberline {1.2}Ziel der Arbeit}{3}{section.1.2}}
|
||||
\@setckpt{chapters/einleitung}{
|
||||
\setcounter{page}{5}
|
||||
\setcounter{page}{4}
|
||||
\setcounter{equation}{0}
|
||||
\setcounter{enumi}{0}
|
||||
\setcounter{enumii}{0}
|
||||
@ -24,17 +24,18 @@
|
||||
\setcounter{figure}{0}
|
||||
\setcounter{table}{0}
|
||||
\setcounter{float@type}{8}
|
||||
\setcounter{Item}{0}
|
||||
\setcounter{Hfootnote}{0}
|
||||
\setcounter{Hy@AnnotLevel}{0}
|
||||
\setcounter{bookmark@seq@number}{5}
|
||||
\setcounter{parentequation}{0}
|
||||
\setcounter{lstnumber}{1}
|
||||
\setcounter{ContinuedFloat}{0}
|
||||
\setcounter{subfigure}{0}
|
||||
\setcounter{subtable}{0}
|
||||
\setcounter{r@tfl@t}{0}
|
||||
\setcounter{Item}{0}
|
||||
\setcounter{Hfootnote}{0}
|
||||
\setcounter{Hy@AnnotLevel}{0}
|
||||
\setcounter{bookmark@seq@number}{5}
|
||||
\setcounter{AM@survey}{0}
|
||||
\setcounter{NAT@ctr}{0}
|
||||
\setcounter{lstlisting}{0}
|
||||
\setcounter{section@level}{0}
|
||||
\setcounter{lstlisting}{0}
|
||||
}
|
||||
|
@ -1,12 +1,17 @@
|
||||
\relax
|
||||
\providecommand\hyper@newdestlabel[2]{}
|
||||
\@writefile{toc}{\contentsline {chapter}{\numberline {4}Ergebnis}{15}{chapter.4}}
|
||||
\@writefile{toc}{\contentsline {chapter}{\numberline {4}Evaluation}{26}{chapter.4}}
|
||||
\@writefile{lof}{\addvspace {10\p@ }}
|
||||
\@writefile{lot}{\addvspace {10\p@ }}
|
||||
\newlabel{ch:ergebnis}{{4}{15}{Ergebnis}{chapter.4}{}}
|
||||
\@writefile{toc}{\contentsline {subsection}{\numberline {4.0.1}Evaluierung}{15}{subsection.4.0.1}}
|
||||
\newlabel{ch:ergebnis}{{4}{26}{Evaluation}{chapter.4}{}}
|
||||
\@writefile{lof}{\contentsline {figure}{\numberline {4.1}{\ignorespaces Vergleich der hochschulinternen E-Mails\relax }}{26}{figure.caption.19}}
|
||||
\@writefile{lof}{\contentsline {figure}{\numberline {4.2}{\ignorespaces Details der relevanten E-Mails des Probanden\relax }}{27}{figure.caption.20}}
|
||||
\@writefile{lof}{\contentsline {figure}{\numberline {4.3}{\ignorespaces Details der irrelevanten E-Mails des Probanden\relax }}{28}{figure.caption.21}}
|
||||
\@writefile{lof}{\contentsline {figure}{\numberline {4.4}{\ignorespaces Vergleich der Anzahl von eintreffenden E-Mails zwischen aktueller Situation und unter Verwendung des Prototyps\relax }}{28}{figure.caption.22}}
|
||||
\@writefile{toc}{\contentsline {subsection}{\numberline {4.0.1}Ergebnis}{29}{subsection.4.0.1}}
|
||||
\@writefile{toc}{\contentsline {subsection}{\numberline {4.0.2}Diskussion}{29}{subsection.4.0.2}}
|
||||
\@setckpt{chapters/ergebnis}{
|
||||
\setcounter{page}{16}
|
||||
\setcounter{page}{31}
|
||||
\setcounter{equation}{0}
|
||||
\setcounter{enumi}{0}
|
||||
\setcounter{enumii}{0}
|
||||
@ -17,24 +22,25 @@
|
||||
\setcounter{part}{0}
|
||||
\setcounter{chapter}{4}
|
||||
\setcounter{section}{0}
|
||||
\setcounter{subsection}{1}
|
||||
\setcounter{subsection}{2}
|
||||
\setcounter{subsubsection}{0}
|
||||
\setcounter{paragraph}{0}
|
||||
\setcounter{subparagraph}{0}
|
||||
\setcounter{figure}{0}
|
||||
\setcounter{figure}{4}
|
||||
\setcounter{table}{0}
|
||||
\setcounter{float@type}{8}
|
||||
\setcounter{Item}{0}
|
||||
\setcounter{Hfootnote}{16}
|
||||
\setcounter{Hy@AnnotLevel}{0}
|
||||
\setcounter{bookmark@seq@number}{26}
|
||||
\setcounter{parentequation}{0}
|
||||
\setcounter{lstnumber}{1}
|
||||
\setcounter{ContinuedFloat}{0}
|
||||
\setcounter{subfigure}{0}
|
||||
\setcounter{subtable}{0}
|
||||
\setcounter{r@tfl@t}{0}
|
||||
\setcounter{Item}{0}
|
||||
\setcounter{Hfootnote}{0}
|
||||
\setcounter{Hy@AnnotLevel}{0}
|
||||
\setcounter{bookmark@seq@number}{21}
|
||||
\setcounter{AM@survey}{0}
|
||||
\setcounter{NAT@ctr}{0}
|
||||
\setcounter{lstlisting}{0}
|
||||
\setcounter{section@level}{0}
|
||||
\setcounter{lstlisting}{0}
|
||||
}
|
||||
|
@ -1,11 +1,13 @@
|
||||
\relax
|
||||
\providecommand\hyper@newdestlabel[2]{}
|
||||
\@writefile{toc}{\contentsline {chapter}{\numberline {5}Zusammenfassung und Ausblick}{17}{chapter.5}}
|
||||
\@writefile{toc}{\contentsline {chapter}{\numberline {5}Schlussbetrachtung}{31}{chapter.5}}
|
||||
\@writefile{lof}{\addvspace {10\p@ }}
|
||||
\@writefile{lot}{\addvspace {10\p@ }}
|
||||
\newlabel{ch:zusammenfassung}{{5}{17}{Zusammenfassung und Ausblick}{chapter.5}{}}
|
||||
\newlabel{ch:zusammenfassung}{{5}{31}{Schlussbetrachtung}{chapter.5}{}}
|
||||
\@writefile{toc}{\contentsline {subsection}{\numberline {5.0.1}Ausblick}{31}{subsection.5.0.1}}
|
||||
\@writefile{lof}{\contentsline {figure}{\numberline {5.1}{\ignorespaces Cron-Tab der im Prototyp getesteten Benachrichtigung.\relax }}{32}{figure.caption.23}}
|
||||
\@setckpt{chapters/fazit}{
|
||||
\setcounter{page}{18}
|
||||
\setcounter{page}{33}
|
||||
\setcounter{equation}{0}
|
||||
\setcounter{enumi}{0}
|
||||
\setcounter{enumii}{0}
|
||||
@ -16,24 +18,25 @@
|
||||
\setcounter{part}{0}
|
||||
\setcounter{chapter}{5}
|
||||
\setcounter{section}{0}
|
||||
\setcounter{subsection}{0}
|
||||
\setcounter{subsection}{1}
|
||||
\setcounter{subsubsection}{0}
|
||||
\setcounter{paragraph}{0}
|
||||
\setcounter{subparagraph}{0}
|
||||
\setcounter{figure}{0}
|
||||
\setcounter{figure}{1}
|
||||
\setcounter{table}{0}
|
||||
\setcounter{float@type}{8}
|
||||
\setcounter{Item}{0}
|
||||
\setcounter{Hfootnote}{16}
|
||||
\setcounter{Hy@AnnotLevel}{0}
|
||||
\setcounter{bookmark@seq@number}{28}
|
||||
\setcounter{parentequation}{0}
|
||||
\setcounter{lstnumber}{1}
|
||||
\setcounter{ContinuedFloat}{0}
|
||||
\setcounter{subfigure}{0}
|
||||
\setcounter{subtable}{0}
|
||||
\setcounter{r@tfl@t}{0}
|
||||
\setcounter{Item}{0}
|
||||
\setcounter{Hfootnote}{0}
|
||||
\setcounter{Hy@AnnotLevel}{0}
|
||||
\setcounter{bookmark@seq@number}{22}
|
||||
\setcounter{AM@survey}{0}
|
||||
\setcounter{NAT@ctr}{0}
|
||||
\setcounter{lstlisting}{0}
|
||||
\setcounter{section@level}{0}
|
||||
\setcounter{lstlisting}{0}
|
||||
}
|
||||
|
@ -1,20 +1,22 @@
|
||||
\relax
|
||||
\providecommand\hyper@newdestlabel[2]{}
|
||||
\@writefile{toc}{\contentsline {chapter}{\numberline {2}Framework}{5}{chapter.2}}
|
||||
\@writefile{toc}{\contentsline {chapter}{\numberline {2}Framework}{4}{chapter.2}}
|
||||
\@writefile{lof}{\addvspace {10\p@ }}
|
||||
\@writefile{lot}{\addvspace {10\p@ }}
|
||||
\newlabel{ch:framework}{{2}{5}{Framework}{chapter.2}{}}
|
||||
\@writefile{toc}{\contentsline {section}{\numberline {2.1}Django}{5}{section.2.1}}
|
||||
\@writefile{lof}{\contentsline {figure}{\numberline {2.1}{\ignorespaces Vereinfachter MVP\relax }}{6}{figure.caption.5}}
|
||||
\@writefile{toc}{\contentsline {subsection}{\numberline {2.1.1}Besonderheiten}{6}{subsection.2.1.1}}
|
||||
\@writefile{lof}{\contentsline {figure}{\numberline {2.2}{\ignorespaces Request-Response-Kreislauf des Django Frameworks\relax }}{7}{figure.caption.6}}
|
||||
\newlabel{ch:framework}{{2}{4}{Framework}{chapter.2}{}}
|
||||
\@writefile{toc}{\contentsline {section}{\numberline {2.1}Django}{4}{section.2.1}}
|
||||
\@writefile{lof}{\contentsline {figure}{\numberline {2.1}{\ignorespaces Vereinfachter MVP [She09]\relax }}{5}{figure.caption.4}}
|
||||
\@writefile{lof}{\contentsline {figure}{\numberline {2.2}{\ignorespaces Request-Response-Kreislauf des Django Frameworks [Nev15]\relax }}{6}{figure.caption.5}}
|
||||
\@writefile{toc}{\contentsline {subsection}{\numberline {2.1.1}Besonderheiten Djangos}{6}{subsection.2.1.1}}
|
||||
\@writefile{toc}{\contentsline {subsection}{\numberline {2.1.2}Virtuelle Umgebung}{7}{subsection.2.1.2}}
|
||||
\@writefile{lof}{\contentsline {figure}{\numberline {2.3}{\ignorespaces Erstellen der virtuelle Umgebung im Terminal\relax }}{7}{figure.caption.6}}
|
||||
\@writefile{toc}{\contentsline {subsection}{\numberline {2.1.3}Sicherheit}{8}{subsection.2.1.3}}
|
||||
\@writefile{toc}{\contentsline {section}{\numberline {2.2}Erweiterungen}{8}{section.2.2}}
|
||||
\@writefile{lof}{\contentsline {figure}{\numberline {2.3}{\ignorespaces Erstellen der virtuelle Umgebung im Terminal\relax }}{8}{figure.caption.7}}
|
||||
\@writefile{lof}{\contentsline {figure}{\numberline {2.4}{\ignorespaces Beispiel eines LDAP-Trees\relax }}{9}{figure.caption.8}}
|
||||
\@writefile{toc}{\contentsline {subsection}{\numberline {2.2.1}Taggable-Manager}{10}{subsection.2.2.1}}
|
||||
\@writefile{toc}{\contentsline {section}{\numberline {2.3}Bootstrap}{11}{section.2.3}}
|
||||
\@writefile{lof}{\contentsline {figure}{\numberline {2.5}{\ignorespaces Einbindung von Bootstrap in einer HTML-Datei\relax }}{11}{figure.caption.9}}
|
||||
\@writefile{lof}{\contentsline {figure}{\numberline {2.6}{\ignorespaces Bootstrap-Klassen in HTML-Tag\relax }}{12}{figure.caption.10}}
|
||||
\@writefile{toc}{\contentsline {subsection}{\numberline {2.2.1}Taggable-Manager}{9}{subsection.2.2.1}}
|
||||
\@writefile{toc}{\contentsline {subsection}{\numberline {2.2.2}Hilfsbibliotheken}{10}{subsection.2.2.2}}
|
||||
\@writefile{toc}{\contentsline {section}{\numberline {2.3}Bootstrap}{10}{section.2.3}}
|
||||
\@writefile{lof}{\contentsline {figure}{\numberline {2.4}{\ignorespaces Einbindung von Bootstrap in einer HTML-Datei\relax }}{11}{figure.caption.7}}
|
||||
\@writefile{lof}{\contentsline {figure}{\numberline {2.5}{\ignorespaces Bootstrap-Klassen in HTML-Tag\relax }}{11}{figure.caption.8}}
|
||||
\@setckpt{chapters/framework}{
|
||||
\setcounter{page}{13}
|
||||
\setcounter{equation}{0}
|
||||
@ -22,7 +24,7 @@
|
||||
\setcounter{enumii}{0}
|
||||
\setcounter{enumiii}{0}
|
||||
\setcounter{enumiv}{0}
|
||||
\setcounter{footnote}{0}
|
||||
\setcounter{footnote}{13}
|
||||
\setcounter{mpfootnote}{0}
|
||||
\setcounter{part}{0}
|
||||
\setcounter{chapter}{2}
|
||||
@ -31,20 +33,21 @@
|
||||
\setcounter{subsubsection}{0}
|
||||
\setcounter{paragraph}{0}
|
||||
\setcounter{subparagraph}{0}
|
||||
\setcounter{figure}{6}
|
||||
\setcounter{figure}{5}
|
||||
\setcounter{table}{0}
|
||||
\setcounter{float@type}{8}
|
||||
\setcounter{Item}{0}
|
||||
\setcounter{Hfootnote}{13}
|
||||
\setcounter{Hy@AnnotLevel}{0}
|
||||
\setcounter{bookmark@seq@number}{14}
|
||||
\setcounter{parentequation}{0}
|
||||
\setcounter{lstnumber}{1}
|
||||
\setcounter{ContinuedFloat}{0}
|
||||
\setcounter{subfigure}{0}
|
||||
\setcounter{subtable}{0}
|
||||
\setcounter{r@tfl@t}{0}
|
||||
\setcounter{Item}{0}
|
||||
\setcounter{Hfootnote}{0}
|
||||
\setcounter{Hy@AnnotLevel}{0}
|
||||
\setcounter{bookmark@seq@number}{11}
|
||||
\setcounter{AM@survey}{0}
|
||||
\setcounter{NAT@ctr}{0}
|
||||
\setcounter{lstlisting}{0}
|
||||
\setcounter{section@level}{0}
|
||||
\setcounter{lstlisting}{0}
|
||||
}
|
||||
|
@ -4,43 +4,55 @@
|
||||
\@writefile{lof}{\addvspace {10\p@ }}
|
||||
\@writefile{lot}{\addvspace {10\p@ }}
|
||||
\newlabel{ch:prototyp}{{3}{13}{Prototyp}{chapter.3}{}}
|
||||
\@writefile{toc}{\contentsline {section}{\numberline {3.1}Organisation}{13}{section.3.1}}
|
||||
\@writefile{toc}{\contentsline {subsection}{\numberline {3.1.1}Verwaltung im Administrator-Backend}{13}{subsection.3.1.1}}
|
||||
\@writefile{toc}{\contentsline {subsection}{\numberline {3.1.2}Berechtigung der User}{13}{subsection.3.1.2}}
|
||||
\@writefile{toc}{\contentsline {section}{\numberline {3.2}Funktion}{13}{section.3.2}}
|
||||
\@writefile{toc}{\contentsline {subsection}{\numberline {3.2.1}Abonnieren}{13}{subsection.3.2.1}}
|
||||
\@writefile{toc}{\contentsline {subsection}{\numberline {3.2.2}Filtern}{13}{subsection.3.2.2}}
|
||||
\@writefile{toc}{\contentsline {subsection}{\numberline {3.2.3}Benachrichtigung}{14}{subsection.3.2.3}}
|
||||
\@writefile{toc}{\contentsline {section}{\numberline {3.1}Forschungsdesign}{13}{section.3.1}}
|
||||
\@writefile{lof}{\contentsline {figure}{\numberline {3.1}{\ignorespaces Forschungsdesign\relax }}{14}{figure.caption.9}}
|
||||
\@writefile{toc}{\contentsline {section}{\numberline {3.2}Organisation}{14}{section.3.2}}
|
||||
\@writefile{toc}{\contentsline {subsection}{\numberline {3.2.1}Datenmodellierung}{15}{subsection.3.2.1}}
|
||||
\@writefile{lof}{\contentsline {figure}{\numberline {3.2}{\ignorespaces CustomUserModel in models.py\relax }}{16}{figure.caption.10}}
|
||||
\@writefile{toc}{\contentsline {subsection}{\numberline {3.2.2}Berechtigungen der User}{16}{subsection.3.2.2}}
|
||||
\@writefile{lof}{\contentsline {figure}{\numberline {3.3}{\ignorespaces Datenmodellierung von \texttt {User} und \texttt {Post}\relax }}{17}{figure.caption.11}}
|
||||
\@writefile{lof}{\contentsline {figure}{\numberline {3.4}{\ignorespaces Menü für eingeloggte Benutzer mit Adminrechte\relax }}{18}{figure.caption.12}}
|
||||
\@writefile{toc}{\contentsline {section}{\numberline {3.3}Funktionen}{18}{section.3.3}}
|
||||
\@writefile{toc}{\contentsline {subsection}{\numberline {3.3.1}Verwaltung der Funktionen}{18}{subsection.3.3.1}}
|
||||
\@writefile{lof}{\contentsline {figure}{\numberline {3.5}{\ignorespaces User Stories\relax }}{19}{figure.caption.13}}
|
||||
\@writefile{toc}{\contentsline {subsection}{\numberline {3.3.2}Artikel abonnieren}{20}{subsection.3.3.2}}
|
||||
\@writefile{lof}{\contentsline {figure}{\numberline {3.6}{\ignorespaces Prototyp Artikel-Editor.\relax }}{21}{figure.caption.14}}
|
||||
\@writefile{lof}{\contentsline {figure}{\numberline {3.7}{\ignorespaces Funktion post\_edit, Auszug aus views.py.\relax }}{21}{figure.caption.15}}
|
||||
\@writefile{lof}{\contentsline {figure}{\numberline {3.8}{\ignorespaces Prototyp Suche- und Abonnier-Seite\relax }}{22}{figure.caption.16}}
|
||||
\@writefile{lof}{\contentsline {figure}{\numberline {3.9}{\ignorespaces Funktion search\_add, Auszug aus views.py.\relax }}{23}{figure.caption.17}}
|
||||
\@writefile{toc}{\contentsline {subsection}{\numberline {3.3.3}Filtern von Artikeln}{23}{subsection.3.3.3}}
|
||||
\@writefile{lof}{\contentsline {figure}{\numberline {3.10}{\ignorespaces Prototyp Newsfeed Seite\relax }}{24}{figure.caption.18}}
|
||||
\@setckpt{chapters/prototyp}{
|
||||
\setcounter{page}{15}
|
||||
\setcounter{page}{26}
|
||||
\setcounter{equation}{0}
|
||||
\setcounter{enumi}{0}
|
||||
\setcounter{enumii}{0}
|
||||
\setcounter{enumiii}{0}
|
||||
\setcounter{enumiv}{0}
|
||||
\setcounter{footnote}{0}
|
||||
\setcounter{footnote}{3}
|
||||
\setcounter{mpfootnote}{0}
|
||||
\setcounter{part}{0}
|
||||
\setcounter{chapter}{3}
|
||||
\setcounter{section}{2}
|
||||
\setcounter{section}{3}
|
||||
\setcounter{subsection}{3}
|
||||
\setcounter{subsubsection}{0}
|
||||
\setcounter{paragraph}{0}
|
||||
\setcounter{subparagraph}{0}
|
||||
\setcounter{figure}{0}
|
||||
\setcounter{figure}{10}
|
||||
\setcounter{table}{0}
|
||||
\setcounter{float@type}{8}
|
||||
\setcounter{Item}{0}
|
||||
\setcounter{Hfootnote}{16}
|
||||
\setcounter{Hy@AnnotLevel}{0}
|
||||
\setcounter{bookmark@seq@number}{23}
|
||||
\setcounter{parentequation}{0}
|
||||
\setcounter{lstnumber}{1}
|
||||
\setcounter{ContinuedFloat}{0}
|
||||
\setcounter{subfigure}{0}
|
||||
\setcounter{subtable}{0}
|
||||
\setcounter{r@tfl@t}{0}
|
||||
\setcounter{Item}{0}
|
||||
\setcounter{Hfootnote}{0}
|
||||
\setcounter{Hy@AnnotLevel}{0}
|
||||
\setcounter{bookmark@seq@number}{19}
|
||||
\setcounter{AM@survey}{0}
|
||||
\setcounter{NAT@ctr}{0}
|
||||
\setcounter{lstlisting}{0}
|
||||
\setcounter{section@level}{0}
|
||||
\setcounter{lstlisting}{0}
|
||||
}
|
||||
|
@ -18,15 +18,16 @@
|
||||
\setcounter{figure}{0}
|
||||
\setcounter{table}{0}
|
||||
\setcounter{float@type}{4}
|
||||
\setcounter{Item}{0}
|
||||
\setcounter{Hfootnote}{0}
|
||||
\setcounter{Hy@AnnotLevel}{0}
|
||||
\setcounter{bookmark@seq@number}{0}
|
||||
\setcounter{parentequation}{0}
|
||||
\setcounter{lstnumber}{1}
|
||||
\setcounter{ContinuedFloat}{0}
|
||||
\setcounter{subfigure}{0}
|
||||
\setcounter{subtable}{0}
|
||||
\setcounter{r@tfl@t}{0}
|
||||
\setcounter{Item}{0}
|
||||
\setcounter{Hfootnote}{0}
|
||||
\setcounter{Hy@AnnotLevel}{0}
|
||||
\setcounter{bookmark@seq@number}{0}
|
||||
\setcounter{AM@survey}{0}
|
||||
\setcounter{NAT@ctr}{0}
|
||||
}
|
||||
|
@ -1,7 +1,7 @@
|
||||
\relax
|
||||
\providecommand\hyper@newdestlabel[2]{}
|
||||
\@setckpt{titlepage/titlepage}{
|
||||
\setcounter{page}{3}
|
||||
\setcounter{page}{1}
|
||||
\setcounter{equation}{0}
|
||||
\setcounter{enumi}{0}
|
||||
\setcounter{enumii}{0}
|
||||
@ -19,17 +19,18 @@
|
||||
\setcounter{figure}{0}
|
||||
\setcounter{table}{0}
|
||||
\setcounter{float@type}{8}
|
||||
\setcounter{Item}{0}
|
||||
\setcounter{Hfootnote}{0}
|
||||
\setcounter{Hy@AnnotLevel}{0}
|
||||
\setcounter{bookmark@seq@number}{0}
|
||||
\setcounter{parentequation}{0}
|
||||
\setcounter{lstnumber}{1}
|
||||
\setcounter{ContinuedFloat}{0}
|
||||
\setcounter{subfigure}{0}
|
||||
\setcounter{subtable}{0}
|
||||
\setcounter{r@tfl@t}{0}
|
||||
\setcounter{Item}{0}
|
||||
\setcounter{Hfootnote}{0}
|
||||
\setcounter{Hy@AnnotLevel}{0}
|
||||
\setcounter{bookmark@seq@number}{0}
|
||||
\setcounter{AM@survey}{0}
|
||||
\setcounter{NAT@ctr}{0}
|
||||
\setcounter{lstlisting}{0}
|
||||
\setcounter{section@level}{0}
|
||||
\setcounter{lstlisting}{0}
|
||||
}
|
||||
|
@ -3,9 +3,13 @@
|
||||
|
||||
\chapter*{Abstract}
|
||||
|
||||
Content of this Bachelor thesis is ....
|
||||
Das Ziel der vorliegenden Bachelor Arbeit ist es, zu beweisen, dass die E-Mail-Flut der Hochschule durch den Einsatz einer Social Media Plattform gedrosselt werden kann. Dazu wird ein Prototyp implementiert der ein personalisierbares Dashboard für Studierende bereitstellt.
|
||||
|
||||
Das Abonnieren von Nachrichten wird mit dem Einsatz von Tags umgesetzt. Benutzer des Systems, versehen die Nachrichten mit Schlagwörtern und machen die Informationen dadurch schneller zuweisbar. Durch die einfach Suche von Tags können Studierende diese nach eigenem Ermessen zum Dashboard hinzufügen.
|
||||
|
||||
Durch den Zähler einer jeden Nachricht können Autoren sehen, ob Informationen bereits gelesen wurden. Hierdurch lassen sich Schlüsse über die Nutzung der Applikation und die Relevanz des Artikels ziehen.
|
||||
|
||||
|
||||
|
||||
|
||||
|
||||
\vspace{\fill}
|
||||
\noindent \textbf{Keywords}: bla,bla Georg Simon Ohm, Wirtschaftsinformatik
|
@ -1,5 +1,5 @@
|
||||
\RequirePackage[ngerman=ngerman-x-latest]{hyphsubst}
|
||||
\documentclass[12pt,a4paper,openright,twoside]{report}
|
||||
\documentclass[12pt,a4paper,openright,oneside]{report}
|
||||
%------------------------------------------------------------------------------
|
||||
% preambule
|
||||
\include{preambule/thesis.preamble}
|
||||
@ -33,13 +33,14 @@
|
||||
\setcounter{page}{1}
|
||||
\pagenumbering{roman}
|
||||
%------------------------------------------------------------------------------
|
||||
% dedication, acknowledgments and quotes
|
||||
\includepdf{AcknowledgmentsDedicationSentence/erklaerung-fertig.pdf}
|
||||
\clearpage{\thispagestyle{empty}}
|
||||
%------------------------------------------------------------------------------
|
||||
% abstract
|
||||
\clearpage{\thispagestyle{empty}}
|
||||
\include{abstract/abstract}
|
||||
%------------------------------------------------------------------------------
|
||||
% dedication, acknowledgments and quotes
|
||||
%\include{AcknowledgmentsDedicationSentence/dedication}
|
||||
%\clearpage{\thispagestyle{empty}}
|
||||
%\include{AcknowledgmentsDedicationSentence/sentence}
|
||||
%\clearpage{\thispagestyle{empty}}
|
||||
%\include{AcknowledgmentsDedicationSentence/acknowledgements}
|
||||
|
@ -1,3 +1,42 @@
|
||||
\chapter{Zusammenfassung und Ausblick}
|
||||
\chapter{Schlussbetrachtung}
|
||||
\label{ch:zusammenfassung}
|
||||
Zusammenfassung...
|
||||
|
||||
Um die Forschungsfrage, \glqq Kann die E-Mail-Flut der Technischen Hochschule mit Hilfe einer Social Media Plattform gedrosselt und die Nachhaltigkeit der Informationen gewährleistet werden?", zu beantworten, wurde in dieser Arbeit ein Prototyp entwickelt.
|
||||
|
||||
Hierbei handelt es sich um eine Erweiterung der Fakultätsinternen Website. Mit Hilfe von Tags können Informationen abonniert und ein personalisiertes Dashboard zusammengestellt werden. Durch differenzierte Berechtigungen ist es realisierbar, bestimmten Benutzern das Einpflegen und Verwalten von Benachrichtigungen zu gewähren.
|
||||
|
||||
In der Evaluation wurde durch eine Kalkulation, stichprobenartig anhand eines Probanden dargelegt, in wie weit die eintreffenden Informationen im Postfach der Studierenden irrelevant sind und durch die individualisierbare Website eingespart werden können.
|
||||
|
||||
Diese wissenschaftliche Arbeit hat gezeigt, dass durch den Einsatz einer webbasierten Plattform, Informationen an verschiedenste Personen einer Einrichtung verteilt werde können. Durch die individuellen Konfigurationsmöglichkeiten sind die Empfänger nicht überfordert und können mit wenig Aufwand passende Themengebiete abonnieren. Die E-Mail-Flut wird stark verkleinert und der administrative Aufwand des Postfachs reduziert.
|
||||
|
||||
Im Folgenden wird beschrieben, wie der Prototyp erweitert und optimiert werden kann.
|
||||
|
||||
|
||||
\subsection{Ausblick}
|
||||
|
||||
Betrachtet wird der aktuelle Stand der Anwendung, wie dieser aufbauend optimiert und in Verwendung gebracht werden kann.
|
||||
|
||||
Der Umfang der verwendeten Tags wird äquivalent zur Dimension der verschiedenen Themen an Informationen stark ansteigen. Es bietet sich an, diese zukünftig in Gruppen zu unterteilen um die Datenbank logisch und performant aufbauen zu können. Zudem ist das automatische Anlegen von Metadaten der einzelnen Tags bei der Administration hilfreich. Beispiele hierfür wären, wann wurde das Schlagwort erstellt, wie oft taucht es in der Suche auf und wie viele Abonnenten besitzt dieses. In Folge dessen käme eine eigens entwickelte Datenkonstruktion für Tags in Frage, um den Taggable Manager abzulösen.
|
||||
|
||||
Eine weitere wichtige Eigenschaft von Informationsplattformen ist das regelmä"sige Abrufen der neusten Mitteilungen. Um das Interesse der Studierenden und Lehrenden aufrecht zu erhalten, kann ein Skript integriert werden, bekannt unter dem Name Cron, dass asynchrone Benachrichtigungen ermöglicht. Wird ein \texttt{Cron-Job} auf dem systeminternen Server ausgeführt, so wird zyklisch der Benutzer über Neuigkeiten per Mail informiert (vgl. [Dok18]).
|
||||
Im Prototyp wurde das Verfahren der asynchronen E-Mail-Benachrichtigung wie folgt getestet: Mit der Konfiguration des Shell-Skripts werden die Sendezyklen und die Inhalte festgelegt. Die Abbildung 5.1. zeigt den Cron-Tab, der wöchentlich gesendet werden soll ( \texttt{0 0 * * 0} ). Des weiteren werden die Pfade der Entwicklungsumgebung und des Servers mitgegeben. Um das Senden der Mails überwachen zu können, werden die Aktivitäten in ein Log-File geschrieben und sind im Administrator-Backend abrufbar. \\
|
||||
|
||||
\begin{figure}[!h]
|
||||
\centering
|
||||
\includegraphics[width=1\textwidth]{figures/crontab}
|
||||
\caption{Cron-Tab der im Prototyp getesteten Benachrichtigung.}
|
||||
\hfill
|
||||
\end{figure}
|
||||
|
||||
Des weiteren ist zur besseren Benutzung der Web-Erweiterung die Usability optimierbar. Die folgenden Punkte zeigen konkrete Möglichkeiten auf, die Gebrauchstauglichkeit für den User zu steigern:
|
||||
|
||||
\begin{itemize}
|
||||
\item Einbindung einer Tagcloud mit Verlinkungen zu Artikeln die diesen Tag enthalten.
|
||||
\item Die Auswahl an Tags für Autoren beim Einpflegen von Artikeln optimieren, zum Beispiel mit Vorschlägen ähnlicher Schlagwörter.
|
||||
\item Das Hinzufügen und Löschen von Tags mit Hilfe einer grafischen Schaltfläche realisieren. Sinnvoll wäre es hierbei die Bibliothek von Bootstrap zu erweitern.
|
||||
\end{itemize}
|
||||
|
||||
Zusammenfassend lässt sich feststellen, dass die Weiterarbeit am Prototyp unabdingbar ist um diesen später in die Fakultätswebsite einbinden zu können.
|
||||
|
||||
|
||||
|
||||
|
@ -1,13 +1,21 @@
|
||||
\chapter{Einleitung}
|
||||
\textcolor{red}{Die vorliegende Arbeit beschäftigt sich mit der wachsenden E-Mail-Flut in den Postfächern der Studierenden und wie man diese reduzieren kann.}
|
||||
Eine Vielzahl an ungelesene E-Mails häufen sich täglich in den Postfächern eines Jeden an (vgl. [Hel12]). Das elektronische Übertragen von Nachrichten ist aus der heutigen Zeit nicht mehr wegzudenken.
|
||||
In der Vergangenheit hat sich jedoch gezeigt, dass das Versenden von Informationen nicht nur Vorteile mit sich bringt. Wie der Spezialist für Gesundheitsprozessberatung in einem Bericht der Mitteldeutschen Zeitung erwähnt, \glqq macht es die stets wachsende E-Mail-Menge unmöglich, sich vernünftig mit den Informationen zu befassen"(vgl. [Ver13]). Nicht nur am Arbeitsplatz, sondern auch an Hochschulen wird Gebrauch davon gemacht, weitere Empfänger oder sogar ganze Verteiler mit in die Kopie einer E-Mail zu integrieren. Hierdurch steigen die irrelevanten Informationen unkontrollierbar schnell an. Infolgedessen nimmt der benötigte Speicheraufwand der zahlreichen kommerziellen, aber auch internen E-Mail-Dienste enorm zu (vgl. [Fio14]).
|
||||
|
||||
Betrachtet man darüber hinaus den Lebenszyklus einer einzelnen E-Mail, wird deutlich, dass dieser nach dem Erstellen, Senden und Empfangen noch nicht abgeschlossen ist. Nachdem die Information vom Adressat geöffnet wurde, wird sie archiviert, muss aber jederzeit durch eine Suchabfrage sofort angezeigt werden können. Dies verdeutlicht den enormen Aufwand, das das Verwalten elektronischer Post mit sich bringt. Aufbauend auf dieser Problematik wird folgend die Ausgangssituation der Arbeit erläutert.
|
||||
|
||||
Schon seit geraumer Zeit ist bekannt, dass das Versenden von Informationen über elektronische Post nicht nur Vorteile mit sich bringt. Wie der Spezialist für Gesundheitsprozessberatung in einem Bericht der Mitteldeutschen Zeitung erwähnt, \glqq macht es die stets wachsende E-Mail-Menge unmöglich sich vernünftig mit den Informationen zu befassen"([Ver13]). Nicht nur am Arbeitsplatz sondern auch in Hochschulen wird Gebrauch gemacht, weitere Empfänger oder sogar ganze Verteiler mit in die Kopie einer E-Mail zu integrieren. Um die Prioritäten der Informationen besser bilden zu können, sollen Studierende selbst entscheiden, welche Nachrichten relevant sind. Hierfür wird eine Social Media Plattform mit personalisierbarem Dashboard implementiert.
|
||||
|
||||
\section{Ausgangssituation}
|
||||
Alle Informationen der Fakultät Elektrotechnik Feinwerktechnik Informationstechnik, kurz efi, werden über die globalen Verteiler des Hochschulinternen Postfaches versendet. Viele dieser Daten sind jedoch nur für eine geringe Schnittmenge der Empfänger relevant und lassen sich nur schwer priorisieren. Das ständig überlastete Postfach muss somit regelmä"sig gepflegt werden. Einen massiven Administrativen Aufwand bedeutet es, E-Mails zu filtern und nach persönlichem Ermessen zu verwalten.
|
||||
Zudem leidet die Nachhaltigkeit der Informationen. Möchten die Empfänger ältere E-Mails abrufen, mussten diese meist schon entfernt werden um Platz für den neuen, eintreffenden E-Mail-Verkehr zu schaffen.
|
||||
Diese Situation führt dazu, dass Empfänger die Informationen meist nicht lesen und sofort entfernen. Die Ersteller haben keinerlei Möglichkeiten zu überprüfen ob und wie viele Studierende und Dozenten eingehende Nachrichten öffnen und lesen.
|
||||
Alle Informationen der Fakultät Elektrotechnik Feinwerktechnik Informationstechnik werden {\"u}ber die globalen Verteiler des Hochschulinternen Postfaches versendet. Viele dieser Daten sind jedoch nur f{\"u}r eine geringe Schnittmenge von Empf{\"a}ngern relevant und enthalten Mitteilungen oder Anhänge, die keinerlei Mehrwert dem Einzelnen aufweisen k{\"o}nnen. Dadurch sind die Postf{\"a}cher der Studierenden schnell überlastet und k{\"o}nnen, ohne regelm{\"a}"sige Pflege, nicht in vollem Umfang genutzt werden. Zudem lassen sich Informationen schwer priorisieren und der massive administrative Aufwand für den Einzelnen, E-Mails selbstständig zu filtern und nach persönlichem Ermessen zu verwalten, steht in keinem Verhältnis zum Mehrwert eines performanten, aufgeräumten Postfaches.
|
||||
|
||||
Die Nachhaltigkeit der Informationen kann meist nicht gewährleistet werden. Grund dafür ist der mangelnde Speicherplatz, verursacht durch die ankommende Nachrichtenflut. M{\"o}chten die Empf{\"a}nger {\"a}ltere E-Mails abrufen, m{\"u}ssen diese meist entfernt werden um Platz f{\"u}r den neuen, eintreffenden E-Mail-Verkehr zu schaffen.
|
||||
Dies kann das Nichtlesen der Informationen seitens der Empfänger verursachen und führt im schlechtesten Fall zum voreiligen Entfernen von relevanten Nachrichten.
|
||||
|
||||
Die Ersteller der Nachrichten haben keine M{\"o}glichkeiten zu {\"u}berpr{\"u}fen, wie viele Studierende eingehende Nachrichten lesen. Eine solche Art der Transparenz wäre jedoch hilfreich, um Informationen inhaltlich zu optimieren und Überschriften treffender zu formulieren.
|
||||
Aus dieser Situation ergibt sich folgende Forschungsfrage: \glqq Kann die E-Mail-Flut der Technischen Hochschule mit Hilfe einer Social Media Plattform gedrosselt und die Nachhaltigkeit der Informationen gewährleistet werden?"
|
||||
|
||||
\section{Ziel der Arbeit}
|
||||
Ziel der Arbeit ist es, durch die Einbindung einer Social Media Plattform den Speicheraufwand des Hochschulpostfaches für Studierende der Efi-Fakultät zu reduzieren. Die Flut an E-Mails soll durch das Verwenden eines personalisierte Dashboard gedrosselt werden. Hierbei wird zunächst der Fokus auf die grundlegenden Funktionen der Website gelegt. Dazu gehört das Abonnieren, einpflegen von neuen und löschen von alten Nachrichten.
|
||||
Zudem sollen die Autoren benachrichtigt werden, in welchem Umfang die hochgeladenen Informationen bereits abonniert und gelesen wurden.
|
||||
Ziel der Arbeit ist es, durch die Einbindung einer Social Media Plattform in die bereits bestehenden Hochschulwebsite den Speicheraufwand des Postfaches f{\"u}r Studierende der Fakultät zu reduzieren. Durch selbständiges Prüfen der Nachrichtenseite nehmen die Zielgruppen Informationen bewusster wahr und können diese individuell an ihre aktuelle Situation anpassen. Das reduziert den administrativen Aufwand und verhindert Speicherengpässe im E-Mail-Postfach. Broadcast ähnliches Senden von Informationen ist hierdurch nur noch in den seltensten Fällen nötig.
|
||||
|
||||
Der Schwerpunkt dieser Arbeit liegt auf der prototypischen Umsetzung der Website-Erweiterung. Hierbei wird zunächst der Fokus auf die grundlegenden Funktionen gelegt. Dazu gehört das Abonnieren, das Einpflegen von neuen und das Löschen von alten Nachrichten. Um den Informationsfluss jedes Einzelnen nicht aus den Augen zu verlieren, soll in regelmä"sigen Abständen eine automatisierte E-Mail verschickt werden.
|
||||
Zudem sollen die Autoren einsehen können, in welchem Umfang die veröffentlichten Informationen bereits gelesen wurden. Dadurch lässt sich nach einer gewissen Zeit feststellen, ob die Studierenden und Dozenten die Nachrichten für relevant erachten und die Plattform weiterhin als verlässliches Portal rentabel ist.
|
||||
|
@ -1,4 +1,83 @@
|
||||
\chapter{Ergebnis}
|
||||
\chapter{Evaluation}
|
||||
\label{ch:ergebnis}
|
||||
\subsection{Evaluierung}
|
||||
|
||||
Ziel dieses Kapitels ist es, zu evaluieren, ob der Prototyp dieser Arbeit die folgende Forschungsfrage beantworten kann:
|
||||
|
||||
\glqq Kann die E-Mail-Flut der Technischen Hochschule mit Hilfe einer Social Media Plattform gedrosselt und die Nachhaltigkeit der Informationen gewährleistet werden?"
|
||||
|
||||
Die Durchführung des Beweises ist realisierbar, indem der E-Mail-Verkehr beispielhaft anhand eines Probanden der Technische Hochschule Nürnberg dargestellt wird. Anschlie"send wird das Ergebnis veranschaulicht.
|
||||
|
||||
Um den Umfang der E-Mail-Flut einordnen zu können, wird das Postfach des Studierenden genauer analysiert. Hierbei handelt es sich um den E-Mail-Verkehr während des Sommersemesters 2018, welches vom 15.03 bis 30.09 andauert. Da die Relevanz der Informationen von den persönlichen Interessen und Aktivitäten des Probanden abhängig ist, kommen diesen Eigenschaften besondere Gewichtung zu. \\
|
||||
|
||||
\begin{figure}[!h]
|
||||
\centering
|
||||
\includegraphics[width=1.0\textwidth]{figures/evaluation-1}
|
||||
\caption{Vergleich der hochschulinternen E-Mails}
|
||||
\hfill
|
||||
\end{figure}
|
||||
|
||||
Während des Semesters trafen in Summe 265 Nachrichten im Postfach des Probanden ein. Darunter sind 127 innerhalb der Fakultät befördert worden. Abbildung 4.1. verdeutlicht auf der linken Seite das Verhältnis zwischen den Nachrichten hochschulweit und innerhalb der Fakultät. Diese wurden von diversen Verteilern an den Probanden gesendet.
|
||||
Auf der rechten Seite der Abbildung 4.1. ist eine Übersicht der fakultätsinternen Verteiler dargestellt. Dabei wird verdeutlicht, dass über die Mailingliste der Studierenden mehr als die Hälfte aller Mitteilungen versendet wird.
|
||||
|
||||
Sortiert der Proband nun den Posteingang nach relevanten Informationen, so zeigt sich folgendes Ergebnis:\\
|
||||
|
||||
\begin{figure}[!h]
|
||||
\centering
|
||||
\includegraphics[width=1.0\textwidth]{figures/evaluation-2}
|
||||
\caption{Details der relevanten E-Mails des Probanden}
|
||||
\hfill
|
||||
\end{figure}
|
||||
|
||||
Das Balkendiagramm der Abbildung 4.2. zeigt, wie viele hochschulweit versendeten Informationen bedeutsam für den Probanden sind. Detaillierter wird gezeigt, wie viele hiervon innerhalb der Fakultät (vgl. intern) von Interesse sind. Das bedeutet, dass 96 Nachrichten im Postfach relevant sind und davon 39 Nachrichten der Fakultät EFI entstammen. Fokussiert man die Themenübersicht wird deutlich, dass allgemeine Benachrichtigungen, wie Termine von Veranstaltungen, prüfungsrelevante Neuigkeiten oder Updates zu den Systemen der Hochschule den Interessenschwerpunkt bilden. Spezifischere Informationen, wie die des Language Centers, des International Office oder der Fachschaft EFI nehmen zwar einen geringeren Anteil ein, sind für den Probanden aber nicht vernachlässigbar.
|
||||
|
||||
Werden die eingehenden E-Mails betrachtet, die der Studierende als nicht relevant aussortiert hat, lassen sich bereits eindeutige Tendenzen erkennen. Wie in Abbildung 4.3. erkennbar ist, sind 169 Informationen, also 64 Prozent der Gesamtheit irrelevant. Extrahiert man davon die fakulätsinternen Benachrichtigungen, so ergibt sich die Anzahl 53. Prozentual lässt sich daraus berechnen, dass etwa 30 Prozent der überflüssigen E-Mails direkt von der EFI-Fakultät ausgehen.
|
||||
|
||||
\begin{figure}[!h]
|
||||
\centering
|
||||
\includegraphics[width=1.0\textwidth]{figures/evaluation-3}
|
||||
\caption{Details der irrelevanten E-Mails des Probanden}
|
||||
\hfill
|
||||
\end{figure}
|
||||
|
||||
Bei Sondierung der Detailansicht auf der linken Seite der Abbildung 4.3., ist zu erkennen, dass meist die sehr spezifischen Informationen über Vorlesungen, Interessen oder Freizeitaktivitäten vom Probanden aussortiert werden. Hierbei lässt sich erschlie"sen, wie symptomatische Informationen trotz fehlender Relevanz, das Postfach überfluten.
|
||||
|
||||
Die beispielhafte Kalkulation der im Postfach des Probanden befindlichen E-Mails wird nun mit dem durch die Implementierung des Prototyps entstehenden Ergebnis verglichen.
|
||||
|
||||
Die Aufgabe der entwickelten Anwendung ist es, die Informationen statt per E-Mail, über eine Social Media Plattform zu publizieren. Dennoch sollen die Studierenden als regelmä"sige Erinnerung eine zusammenfassende Benachrichtigung erhalten. Stellt man die Anzahl, der über das Semester verteilten eintreffenden E-Mails des Probanden, den wöchentlichen Mitteilungen des Prototypen gegenüber, so ergibt sich folgendes Diagramm:\\
|
||||
|
||||
\begin{figure}[!h]
|
||||
\centering
|
||||
\includegraphics[width=0.4\textwidth]{figures/evaluation-5}
|
||||
\caption{Vergleich der Anzahl von eintreffenden E-Mails zwischen aktueller Situation und unter Verwendung des Prototyps}
|
||||
\hfill
|
||||
\end{figure}
|
||||
|
||||
Abbildung 4.4. zeigt deutlich, dass durch den Einsatz des Prototyps die Anzahl der Informationen im Postfach stark reduziert werden können. Der Proband würde etwa 90 Prozent seines Nachrichtenvolumens im System einsparen. Dadurch bestätigt sich die Hypothese: Die E-Mail-Flut der Hochschule wird durch den Einsatz einer Weberweiterung reduziert.
|
||||
|
||||
|
||||
\subsection{Ergebnis}
|
||||
|
||||
Werden alle Auswertungen der Evaluation zusammengefasst und betrachtet, so ist deutlich zu sehen, dass Benachrichtigungen der Hochschule zu ausgedehnt verteilt werden. Fakultätsübergreifende Themengebiete sind häufig über umfangreiche Verteiler an Einzelpersonen weitergegeben worden und erzeugen dabei eine schwer administrierbare Menge.
|
||||
|
||||
Der Fokus dieser Arbeit liegt auf dem reduzieren der E-Mail-Flut innerhalb der EFI-Fakultät. Wird der Prototyp auf der Hochschul-Website eingebunden, so kann die Problematik im Idealfall auf ein Kleinstes reduziert werden. Das Verhältnis zwischen irrelevanten und relevanten Informationen zeigt deutlich, dass über die Hälfte der Nachrichten keinerlei Bedeutsamkeit für den Probanden haben. Aufgrund dessen, lassen sich folgende Erkenntnisse festhalten. Die Website-Erweiterung vermeidet das Eintreffen der unwichtigen E-Mails und informiert Studierende über alle wichtigen Benachrichtigungen.
|
||||
|
||||
Werden die Informationen der gesamten Hochschule in das System eingetragen, so kann das Postfach lediglich für persönliche und organisatorische Absprachen innerhalb der Hochschule genutzt werden und der administrative Aufwand des E-Mail-Speichers kann aufs Kleinste beschränkt werden.
|
||||
|
||||
|
||||
\subsection{Diskussion}
|
||||
In diesem Kapitel wird das Ergebnis der Arbeit in Bezug auf die Forschungsfrage diskutiert. Au"serdem wird der Prototyp mit einem bereits vorhandenen Framework verglichen und in Bezug darauf eingeordnet.
|
||||
|
||||
Unter Verwendung der entwickelten Erweiterung kann die E-Mail-Flut der Hochschule gedrosselt werden. Die Evaluation anhand eines Probanden zeigt eindeutig das Potenzial, durch eine optimierte Personalisierbarkeit, die Anzahl von Nachrichten zu reduzieren. Anhand der beispielhaften Zählung der im Postfach vorhandenen E-Mails kann zudem festgehalten werden, dass eine Gro"szahl dieser als unnötig für Individuen einzustufen ist. Zu beachten ist jedoch, dass es sich bei der Bewertung nur um eine theoretische Annäherung eines realen Ergebnisses handelt.
|
||||
|
||||
Weitere Schritte, um den Einsatz des Prototypen finalisieren zu können, sind ein ausführliches Testing, was im Rahmen dieser Arbeit nicht untersucht wurde. Unter Beobachtung der einzubindenden Web-Erweiterung kann die Plattform für einen definierten Zeitraum genutzt werden und in Folge dessen eine detaillierte Aussage über die mögliche Reduzierung des Speicheraufwands im Postfach möglich sein.
|
||||
|
||||
Das Ergebnis dieser Arbeit wird im Folgenden mit Eigenschaften des Kursmanagementsystems Moodle verglichen.
|
||||
Die Struktur des Prototypen ist, wie in den oberen Kapiteln bereits erläutert, mit verschieden einordenbaren Tags realisiert. Bestimmte Benutzer können Informationen einpflegen und verwalten. In Moodle ist der Vorgang ähnlich realisiert. Hier können Lehrende Material und Informationen in verschiedenen Lernräumen hochladen. Das System ist, im Gegensatz zum Prototyp, sehr umfangreich. Als Benutzer ist es möglich, sich in diese Lernräume einzutragen und durch die Anmeldung aktuelle Benachrichtigungen zu erhalten. Sind manche Informationen in Moodle nur mit einem extra Passwort zugänglich, so ist das in der Erweiterung dieser Arbeit für alle Benutzer gleich verfügbar (vgl. [Dok15]).
|
||||
|
||||
Die Menge der Daten einer solchen Plattform sind nicht zu unterschätzen. Moodle verwendet unter anderem Caching-Tools und optimierte Prozesse, um die Datenbanken zu befüllen und schnellstmöglich abfragen zu können. In dieser Arbeit liegt der Schwerpunkt hingegen nicht auf der Optimierung einer Datenbank oder dem Verbessern der Performanz.
|
||||
|
||||
Zusammenfassend lässt sich sagen, dass Moodle ein zu umfangreiches Repertoire an Möglichkeiten bietet, im Gegensatz zum entwickelten Prototyp. Alle Funktionen, die in der Web-Erweiterung enthalten sind, lassen darauf schlie"sen, dass unter Betrachtung der Evaluation, der Prototyp zur Reduzierung der E-Mail-Flut beitragen kann.
|
||||
|
||||
|
||||
|
||||
|
||||
|
@ -1,65 +1,57 @@
|
||||
\chapter{Framework}
|
||||
\label{ch:framework}
|
||||
Um die Website-Erweiterung realisieren zu können, wird zunächst festgelegt welche Programmierschnittstellen verwendet werden. Im Web-Backend fällt die Wahl auf die objektorientierte Sprache Python, die ausschlie"slich Serverseitig anwendbar ist. Der Programmaufbau Pythons macht den Code leicht lesbar und der einfache Syntax ermöglicht einen strukturierte Implementierung der Website([Ndu17]). Die vielen abstrakten Datentypen, wie dynamische Arrays und Wörterbücher, sind gro"sflächig einsetzbar.
|
||||
Ein entscheidender Vorteil hierbei ist das dazugehörige Framework Django, auf das im folgenden Kapitel genauer eingegangen wird.
|
||||
|
||||
Um die Website-Erweiterung realisieren zu können, wird zunächst festgelegt, welche Programmierschnittstellen verwendet werden. Im Web-Backend fällt die Wahl auf die objektorientierte Sprache Python, die serverseitig anwendbar ist. Der Programmaufbau Pythons macht den Code leicht lesbar und die einfache Syntax ermöglicht eine strukturierte Implementierung der Website (vgl. [Ndu17]). Durch den modularen Aufbau ist es selbst für unerfahrene Entwickler möglich, die Sprache schnell zu erlernen. Darüber hinaus bringt Python verschiedene Web-Service Tools mit sich, die beim Implementieren einer Website viel Zeit sparen und das Aneignen von komplexen Protokollen redundant machen (vgl. [Sol17]).
|
||||
Das dazugehörige Framework Django reduziert den Entwicklungsaufwand eines Prototypen erheblich und ist daher als zielführendes Framework die beste Wahl.
|
||||
|
||||
\section{Django}
|
||||
Django ist ein Web-Framework, das auf einer Model-View-Presenter (MVP) Architektur basiert. Ähnlich wie der Model-View-Controller sind die Interaktionen zwischen Model und View die Auswahl und Ausführung von Befehlen und das Auslösen von Ereignissen (vgl. Abbildung 2.1). Da die View aber hier bereits den Gro"steil des Controllers übernimmt, ist der MVP eine Überarbeitung. Der Teil, der Elemente des Modells auswählt, Operationen durchführt und alle Ereignisse kapselt, ergibt die Presenter-Klasse([She09]). Durch die direkte Bindung von Daten und View, geregelt durch den Presenter, wird die Codemenge der Applikation stark reduziert.
|
||||
Django ist ein Web-Framework, das eine schnelle, strukturierte Entwicklung ermöglicht und dabei ein einfaches Design beibehält. Der darin enthaltene Model-View-Presenter (MVP) kann, ähnlich wie der Model-View-Controller, die Interaktionen zwischen Model und View, die Auswahl und Ausführung von Befehlen und das Auslösen von Ereignissen steuern (vgl. Abbildung 2.1). Da die View aber hier bereits den Gro"steil des Controllers übernimmt, ist der MVP eine Überarbeitung. Der Teil, der Elemente des Modells auswählt, Operationen durchführt und alle Ereignisse kapselt, ergibt die Presenter-Klasse (vgl. [She09]). Durch die direkte Bindung von Daten und der View, geregelt durch den Presenter, wird die Codemenge der Applikation stark reduziert.
|
||||
|
||||
\begin{figure}[!h]
|
||||
\centering
|
||||
\includegraphics[width=0.5\textwidth]{figures/MVP}
|
||||
\caption{Vereinfachter MVP}
|
||||
\includegraphics[width=0.6\textwidth]{figures/MVP}
|
||||
\caption{Vereinfachter MVP [She09]}
|
||||
\hfill
|
||||
\end{figure}
|
||||
|
||||
Der Prozess vom Anfragen der URL über den Server, bis hin zur fertig gerenderten Website kann wie folgt vereinfacht darstellen. Der User gibt eine URL im Browser ein und sendet sie an den Web-Server.
|
||||
Der Prozess vom Anfragen der URL über den Server bis hin zur fertig gerenderten Website kann wie folgt vereinfacht dargestellt werden.
|
||||
|
||||
Das Interface WSGI am Web-Server verbindet diesen mit dem Web-Framework, indem es den Request zum passenden Objekt weiterleitet. Hier wird der Applikation eine Callback-Funktion zur Verfügung gestellt [Kin17]. Au"serdem werden folgende Schritte durchgeführt:
|
||||
Der User gibt eine URL im Browser ein und sendet sie an den Web-Server. Das Interface WSGI (Web Server Gateway Interface) am Web-Server verbindet diesen mit dem Web-Framework, indem es den Request zum passenden Objekt weiterleitet. Hier wird der Applikation eine Callback-Funktion zur Verfügung gestellt (vgl. [Kin17]). Au"serdem werden folgende Schritte durchgeführt:
|
||||
\begin{itemize}
|
||||
\item Die Middleware-Klassen aus der settings.py werden geladen
|
||||
\item Die Middleware-Klassen aus der \texttt{settings.py} werden geladen
|
||||
\item Die Methoden der Listen Request, View, Response und Excpetion werden geladen
|
||||
\item Die angeforderte URL wird aufgelöst
|
||||
\end{itemize}
|
||||
|
||||
Der WSGI-Handler fungiert also als Pförtner und Manager zwischen dem Web-Server und dem Django-Projekt.
|
||||
Um die URL, wie weiter oben erwähnt, aufzulösen, benötigt WSGI einen urlresolver. Durch die explizite Zuweisung der vorhandenen Seiten, kann dieser über die regulären Ausdrücke der url.py-Datei iterieren. Gibt es eine Übereinstimmung, wird die damit verknüpfte Funktion in der View (view.py) aufgerufen. Hier ist die gesamte Logik der Website lokalisiert. Wie bereits erwähnt, ist es möglich unter Anderem auf die Datenbank der Applikation zuzugreifen und Eingaben des Users über eine Form zu verarbeiten. Nachdem werden die Informationen der View an das Template weitergereicht. Es handelt sich dabei um eine einfache HTML-Seite in der der strukturelle Aufbau im Frontend festgelegt wird. Die Informationen der View können hier zwischen doppelt-geschweiften Klammern eingebunden und, wenn nötig, mit einfachen Python-Befehlen angepasst werden. Nun kann das Template, die vom WSGI-Framework zur Verfügung gestellte Callback-Funktion befüllen und einen Response an den Web-Server schicken. Die fertige Seite ist beim Klienten im Browserfenster zum rendern bereit (vgl. Abbildung 2.2.).
|
||||
Der WSGI-Handler fungiert als Pförtner und Manager zwischen dem Web-Server und dem Django-Projekt.
|
||||
Um die URL aufzulösen, benötigt WSGI einen \textit {urlresolver}\footnote{ Urlsresolver verknüpft Url-Muster mit den passenden Views.}
|
||||
(vgl. [Dja18b]). Durch die explizite Zuweisung der vorhandenen Seiten kann dieser über die regulären Ausdrücke der \texttt{url.py}-Datei iterieren. Gibt es eine Übereinstimmung, wird die damit verknüpfte Funktion in der View (\texttt{views.py}) aufgerufen. Hier ist die gesamte Logik der Website lokalisiert. Unter anderem ist es möglich auf die Datenbank der Applikation zuzugreifen und Eingaben des Users über eine Form zu verarbeiten. Anschlie"send werden die Informationen der View an das Template weitergereicht. Es handelt sich dabei um eine einfache HTML-Seite, in der der strukturelle Aufbau im Frontend festgelegt wird. Die Informationen der View können hier zwischen doppelt-geschweiften Klammern eingebunden und, wenn nötig, mit einfachen Python-Befehlen angepasst werden. Das Template kann dann die vom WSGI-Framework zur Verfügung gestellte Callback-Funktion befüllen und eine Response an den Web-Server schicken. Die fertige Seite ist beim Klienten im Browserfenster zum Rendern bereit (vgl. [Kin17], Abbildung 2.2.).
|
||||
|
||||
\begin{figure}[!h]
|
||||
\centering
|
||||
\includegraphics[width=0.5\textwidth]{figures/request-response-cycle}
|
||||
\caption{Request-Response-Kreislauf des Django Frameworks}
|
||||
\caption{Request-Response-Kreislauf des Django Frameworks [Nev15]}
|
||||
\hfill
|
||||
\end{figure}
|
||||
|
||||
\subsection {Besonderheiten}
|
||||
Das Django-Framework bringt einige Besonderheiten mit sich, die beim implementiern des Prototypen von Bedeutung sind. Diese werden im Folgenden beschrieben.
|
||||
\subsection {Besonderheiten Djangos}
|
||||
Das Django-Framework bringt einige Besonderheiten mit sich, die beim Implementieren des Prototypen von Bedeutung sind. Diese werden im Folgenden beschrieben.
|
||||
|
||||
Die Administratoroberfläche ist eines der hilfreichsten Werkzeugen des gesamten Frameworks. Es stellt die Metadaten der Modelle aus dem Code visuell dar. Verifizierte Benutzer können die Daten nicht nur schnell erfassen, sondern diese auch editieren und verwalten. Das Recht, das Admin-Backend uneingeschränkt zu benutzen, ist dem sogenannten superuser vorenthalten. Dieser kann beim erstmaligen zuweisen nur über die Kommandozeile eingerichtet werden. Ist bereits ein superuser vorhanden, kann dieser im Admin-Backend weiteren Benutzern den gleichen Handlungsfreiraum einräumen. Zudem gibt es noch weitere Stufen der Zugangsberechtigungen, Staff- und Active-Status, die für eine breitere Gruppe von Benutzern geeignet ist.
|
||||
Um die gestaffelten Zugangsberechtigungen auch auf der Website umsetzen zu können, stellt Django verschiedene Decorator zur Verfügung. Soll eine bestimmte Seite nur von eingeloggten Benutzern besucht werden dürfen, so importiert man die Decorator des, von Django zur Verfügung gestellten, Authentifizierungssystems mit
|
||||
Die Administratoroberfläche ist eines der hilfreichsten Werkzeugen des gesamten Frameworks. Es stellt die Metadaten der Modelle aus dem Code visuell dar. Verifizierte Benutzer können die Daten nicht nur schnell erfassen, sondern diese auch editieren und verwalten. Das Recht, das Admin-Backend uneingeschränkt zu nutzen, ist dem \textit{Superuser}\footnote{ Superuser ist ein Benutzer, der alle Zugangsrechte im Framework erhält.} vorenthalten. Dieser kann beim erstmaligen Zuweisen nur über die Kommandozeile eingerichtet werden. Ist bereits ein Superuser vorhanden, kann dieser im Admin-Backend weiteren Benutzern den gleichen Handlungsfreiraum einräumen. Zudem gibt es noch weitere Stufen der Zugangsberechtigungen, Staff- und Active-Status, die für eine breitere Gruppe von Benutzern geeignet ist.
|
||||
Um die gestaffelten Zugangsberechtigungen auch auf der Website umsetzen zu können, stellt Django verschiedene Dekoratoren zur Verfügung. Soll eine bestimmte Seite nur von authentifizierten Benutzern besucht werden können, kann die Funktion mit einem Dekorator versehen werden:
|
||||
\\
|
||||
\begin{addmargin}[0pt]{0pt}
|
||||
\noindent\hspace*{10mm}%
|
||||
from django.contrib.auth.decorators import login\_required
|
||||
\texttt{\makeatletter @login\_required}
|
||||
\\
|
||||
Direkt über den Beginn der Funktion in view.py, oder auch single-view-function, wird zudem folgende Zeile ergänzt:
|
||||
\\
|
||||
\noindent\hspace*{10mm}%
|
||||
\makeatletter @login\_required
|
||||
\\
|
||||
Natürlich lassen sich die Decoratoren auch für andere Zwecke vor Funktionen platzieren. Begrenzte Zugänge zu den Views können durch die Art der Anfrage realisiert werden. Der Benutzer muss also zum Beispiel durch GET auf eine Seite zugreifen wollen um Einsicht zu erhalten.
|
||||
\end{addmargin}
|
||||
Natürlich lassen sich Dekoratoren auch für andere Zwecke vor Funktionen platzieren. Unter anderem werden so die Views vor möglichen Angriffen, wie \textit{Cross-Site-Scripting}\footnote{ Cross-Site-Scripting kann Webseiten verändern oder Passwörter abgreifen.}, geschützt.
|
||||
|
||||
Benutzerdefinierte Decorator sind im Django-Framework möglich, darauf wird hier aber nicht weiter eingegangen.
|
||||
Durch den einfachen Aufbau ist es au"serdem möglich, Dekoratoren selbst zu implementieren. Ein einfaches Beispiel wäre das Prüfen des an die Funktion übergebenen Parameter. Sollen nur positive Zahlen verarbeitet werden, so kann der Decorator alle anderen Eingaben abfangen.
|
||||
|
||||
|
||||
\subsection{Virtuelle Umgebung}
|
||||
|
||||
|
||||
|
||||
\section{Erweiterungen}
|
||||
|
||||
Die Programmiersprache Python bringt viele hilfreiche Erweiterungen mit sich. Die im folgenden aufgeführten sind hauptsächlich für die Paketverwaltung der Applikation zuständig.
|
||||
|
||||
Wird ein neues Projekt gestartet, ist es üblich, verschiedensten Programme und Module dafür zu verwenden. Die Versionen dieser spielen hierbei eine entscheidende Rolle, um Konflikte zu vermeiden [Fou18]. Um diesem Problem vorzubeugen, wird eine virtuelle Umgebung implementiert. Diese besitzt einen eigenen Projektpfad, beinhaltet alle nötigen Pakete und Bibliotheken, und lässt sich nach dem Einrichten im Terminal benutzten. Die folgende Abbildung (2.3) zeigt das Erstellen eines neunen Ordners, das Erstellen der virtuellen Umgebung und den Aktivierungsbefehl. Ist der Name des Environment in Klammern am Anfang der Zeile, bedeutet das, diese ist jetzt aktiv.
|
||||
Wird eine prototypische Anwendung gestartet, ist es notwendig, verschiedenste Module zu verwenden und zu testen. Die Versionen dieser Module spielen hierbei eine entscheidende Rolle, um Konflikte zu vermeiden (vgl. [Fou18]). Um diesem Problem vorzubeugen, wird eine virtuelle Umgebung implementiert. Sie besitzt einen eigenen Projektpfad, beinhaltet alle nötigen Pakete und Bibliotheken, und lässt sich nach dem Einrichten im Terminal starten. Abbildung 2.3. zeigt das Erstellen eines neuen Ordners, das Erstellen der virtuellen Umgebung und den Aktivierungsbefehl. Ist der Name des Environments in Klammern am Kommandozeilenanfang, bedeutet das, dass diese jetzt aktiv ist.
|
||||
|
||||
\begin{figure}[!h]
|
||||
\centering
|
||||
@ -68,57 +60,54 @@ Wird ein neues Projekt gestartet, ist es üblich, verschiedensten Programme und
|
||||
\hfill
|
||||
\end{figure}
|
||||
|
||||
Die Umgebung ist jetzt bereit befüllt zu werden. Um die Pakete und Module kollisionsfrei zu installieren ist es empfehlenswert einen Package-Manager zu verwenden. Mit pip können diese installiert, geupdated und gelöscht werden. Au"serdem kann der Manager Abhängigkeiten, wenn nötig, überschreiben und optimieren. Falls ein, sich von der neuesten Version unterscheidendes, Programm installiert werden soll, so ist dies ebenso möglich.
|
||||
Um die Pakete und Module kollisionsfrei zu installieren, ist es empfehlenswert, einen Package-Manager zu verwenden. Mit \textit{pip}\footnote{ pip ist das rekursive Akronym für pip Install Packages. } (vgl. [Wei17, K. 23.1]) können diese installiert, geupdated und gelöscht werden. Au"serdem kann der Manager Abhängigkeiten, wenn nötig, überschreiben und optimieren. Falls ein sich von der neuesten Version unterscheidendes Programm installiert werden soll, so ist dies ebenso möglich.
|
||||
|
||||
Eine weitere hilfreiche Erweiterung ist pylint. Das Tool sucht nicht nur nach Fehlern im Code, sondern versucht diesen sauber und einheitlich zu gestalten. Hierbei wird auf den Code-Standard PEP-8 geprüft [Dix18]. Die folgende Liste zeigt eine Kurzfassung der wichtigsten Regeln:
|
||||
\subsection{Sicherheit}
|
||||
Beim Implementieren einer Website ist das Absichern vor schädlichen Attacken heutzutage unabdingbar. Django aktiviert einige Funktionen zum Schutz bereits beim Projektstart automatisch. Dazu gehört das Abwehren von \textit{Cross-Site-Scripting}, \textit{SQL-Injektion}\footnote{ SQL-Injection nutzt mangelnde Überprüfung von Metazeichen um Datenbankabfragen zu verändern.}, \textit{Clickjacking}\footnote{ Clickjacking ist eine Überlagerung von Internetseiten um Klicks zu manipulieren.}
|
||||
und die Sicherstellung der \textit{Session-Security}\footnote{ Session-Security setzen Cookies auch für Subdomäne beim Client. }.
|
||||
|
||||
\begin{itemize}
|
||||
\item Einrückung, meist 4 Leerzeichen
|
||||
\item Maximale Zeichenanzahl pro Zeile
|
||||
\item Zwei Leerzeile zwischen Klassen und Funktionen
|
||||
\item Eine Leerzeile zwischen Methoden innerhalb einer Klasse
|
||||
\item Leerzeichen in Ausdrücke und Anweisungen vermeiden
|
||||
\item Die Reihenfolge der Importe: Standartbibliotheken, Drittanbieterbibliotheken, Lokale Anwendungen
|
||||
\item Konventionen der Namensgebung von Funktionen, Modulen usw.
|
||||
\end{itemize}
|
||||
Natürlich sind dies Vorgaben, die eingehalten werden können, aber nicht notwendig sind um den Code fertig kompilieren und ausgeben zu lassen.
|
||||
Sollen die Formulare des Prototypen gegen \textit{Cross-Site-Request-Forgery} geschützt werden, muss aktiv ein \textit{Token} im Template gesetzt werden.
|
||||
|
||||
Das ldap, Lightweight Directory Access Protocol, muss als Erweiterung in die hier bearbeitende Bachelor-Arbeit eingebunden werden. Dies ist ein Internetprotokoll, welches die Kommunikation mit dem Hochschulinternen Active Directory verwaltet. Es wird eingesetzt um Benutzer so schnell und effizient wie möglich durch eine bereits existierende Datenbank abzufragen und zu authentifizieren. Der Aufbau ist mit einem Baum zu vergleichen (vgl. Abbildung 2.4.). Die Wurzel besteht aus sehr allgemeinen Informationen, umso näher man den Blättern kommt, umso spezifischer werden diese. Ein Objekt in der Struktur wird durch einen einmaligen Namen identifiziert, der aus den gesamten hinterlegten Informationen besteht. Der Name für den in der Abbildung 2.4 dargestellten Baum wäre \glqq cn=John Doe, ou=People, dc=sun.com\grqq.
|
||||
Ein solcher Angriff tritt auf, wenn über einen bösartigen Link, eine Formularschaltfläche oder einfach den eingebettetem JavaScript-Code die Daten im Server verändert werden sollen. Hierbei nutzt der Angreifer die Rechte eines eingeloggten Benutzers und kann somit Informationen im Backend verfälschen. Um dies zu verhindern, wird im Template des Prototypen zwischen den Form-Tags der \textit{crsf-Token}\footnote{ Token sind Komponente die eine Zugriffskontrolle von Benutzer druchführen können.}
|
||||
eingefügt. Der Token setzt einen Cookie mit einer verschlüsselten Zufallszahl. Das Gleiche passiert im Template, wo ein für den Benutzer nicht sichtbares Form-Feld die gleiche verschlüsselte Zufallszahl erhält. Beide Zahlen erhalten zudem einen \textit{Salt}, einen generierten Zusatzwert, der das Entschlüsseln dieser um ein vielfaches erschwert. Wird ein Request gesendet, vergleicht eine von Django initialisierte \textit{Middleware}\footnote{ Middleware ist ein Plug-in, dass Anfrage- und Antwortverarbeitung durchführt.} beide Zahlen. Stimmen diese nicht überein, hat also ein Dritter die Informationen manipuliert, wird der 403 HTTP-Standard-Statuscode gesendet, welcher besagt, dass der Server eine Anfrage erhalten hat, diese aber nicht erfüllen wird.
|
||||
Der \texttt{crsf-Token} greift nur, wenn der POST-Request innerhalb der eigenen Website gesendet wird und nicht über URLs, die au"serhalb des Frameworks liegen (vgl. [Fou18a]).
|
||||
|
||||
\begin{figure}[!h]
|
||||
\centering
|
||||
\includegraphics[width=0.5\textwidth]{figures/ldap-tree}
|
||||
\caption{Beispiel eines LDAP-Trees}
|
||||
\hfill
|
||||
\end{figure}
|
||||
\section {Erweiterungen}
|
||||
Django bringt viele hilfreiche Erweiterungen mit sich, die mit einem Packagemanager einfach in die virtuelle Umgebung geladen werden können. Um das passende Add-on für ein Projekt zu finden, bietet die Plattform djangopackages.org alle Erweiterungen in übersichtlichen Tabellen mit Eigenschaften und Bewertungen an. Eine Vielzahl an Bibliotheken wurden für diese Arbeit getestet, aber wegen mangelnder Kompatibilität oder Funktionalität als nicht hilfreich erachtet und deshalb hier nicht weiter erwähnt.
|
||||
Die im Folgenden aufgeführten Bibliotheken sind im Prototyp zur Anwendung gelangt.
|
||||
|
||||
\subsection {Taggable-Manager}
|
||||
Django-taggit ist eine Erweiterung von Alex Gaynor, einem Entwickler aus Washingtion DC. Das Add-on ermöglicht das Verwenden von Tags die automatisch mit einem eindeutigen Slug versehen werden. Der darin enthaltene Taggable Manager verwendet Django's Contenttype Framework, welches per Default verwendet wird, um die Modelle der Applikation zu verfolgen und diese durch generische Beziehungen zu verknüpfen. Die Felder app\_label und model machen die Modelle eindeutig zuweisbar. Instanzen des Contenttypes präsentieren und speichern die Informationen und Erstellen automatisch neue Instanzen, wenn Modelle hinzugefügt werden. Zudem stehen Methoden zur Verfügung, die das Abrufen und Arbeiten mit Instanzen der einzelnen Modelle erleichtern.
|
||||
Um die Artikel besser priorisieren und die Informationsflut für Benutzer reduzieren zu können, wird im Prototyp mit Schlagwörtern gearbeitet. Django-taggit ist eine Erweiterung, die das Verwenden von Tags vereinfacht. Der darin enthaltene Taggable Manager setzt Django's Contenttype Framework ein, welches per Default verwendet wird, um die Modelle der Applikation zu verfolgen und diese durch generische Beziehungen zu verknüpfen. Die Felder \texttt{app\_label} und \texttt{model} machen die Modelle eindeutig zuweisbar. Instanzen des Contenttyps präsentieren und speichern die Informationen und erstellen automatisch neue Instanzen, wenn Modelle hinzugefügt werden. Zudem stehen Methoden zur Verfügung, die das Abrufen und Arbeiten mit Instanzen der einzelnen Modelle erleichtern.
|
||||
|
||||
Der Taggable-Manager ist jedoch nicht an das Contenttype-Framework gebunden ([Her16]). Durch die Verwendung eines echten Fremdschlüssels, kann zum Beispiel die Performance und Referenzgarantie verwirklicht werden. Dazu ist lediglich die Erstellung einer eigenen lookup-Tabelle notwendig, die die Entitäten zweier Tabellen direkt verlinkt, anstatt diese generische zu verbinden. Weiterführend können Modelle in einem benutzerdefinierten Modell vereint werden, sodass er Zugriff auf diese einheitlich geschieht. Außerdem ist es möglich Primary-Keys zu verwenden, die nicht aus ganzen Zahlen bestehen, sondern aus Buchstaben und Wörtern.
|
||||
Der Taggable-Manager ist jedoch nicht an das Contenttype-Framework gebunden (vgl. [Her16]). Durch die Verwendung eines echten Fremdschlüssels kann zum Beispiel die Performance und Referenzgarantie verwirklicht werden. Dazu ist lediglich die Erstellung einer eigenen \textit{Lookup-Tabelle}\footnote{ Eine Lookup-Tabelle speichert Daten statisch im Verhältnis zueinander.} notwendig, die die Entitäten zweier Tabellen direkt verlinkt, statt sie generisch zu verbinden. Weiterführend können Modelle in einem benutzerdefinierten Modell vereint werden, sodass der Zugriff darauf einheitlich geschieht. Au"serdem ist es möglich, Primary-Keys zu verwenden, die nicht aus ganzen Zahlen bestehen, sondern aus Buchstaben und Wörtern.
|
||||
|
||||
Um django-taggit zu installieren wird der folgende Befehl in die Kommandozeile eingefügt([Gay10]):
|
||||
Um django-taggit zu installieren, wird der folgende Befehl in die Kommandozeile eingefügt (vgl. [Gay10]):
|
||||
\\
|
||||
\begin{addmargin}[0pt]{0pt}
|
||||
\noindent\hspace*{10mm}%
|
||||
\$ pip install django-taggit
|
||||
\texttt{\$ pip install django-taggit }
|
||||
\\
|
||||
Im model.py wird das Feld tag neu erstellt und als Taggable Manager definiert.
|
||||
Au"serdem muss taggit in der settings.py unter INSTALLED\_APPS ergänzt werden. Um dem Programm zu sagen, dass nun eine neue Liste der Datenbank hinzugefügt werden muss, werden folgende Befehle in die Kommandozeile eingefügt:
|
||||
\\
|
||||
\noindent\hspace*{10mm}%
|
||||
\$ python3 manage.py makemigrations \\
|
||||
\noindent\hspace*{10mm}%
|
||||
\$ python3 manage.py migrate
|
||||
\\
|
||||
Im Admin-Backend kann nun geprüft werden, ob das neue Feld in die Datenbank integriert wurde. Neue Tags können in das Textfeld eingetragen werden. Der Parser verarbeitet jedes Wort, dass durch ein Komma oder ein Leerzeichen getrennt ist als ein Tag. Soll dieses jedoch aus mehreren Wörtern bestehen so müssen diese mit Anführungszeichen umfasst werden. Standardmäßig unterscheidet der Taggable Manager zwischen Groß- und Kleinschreibung, Tags sind also case sensitive. Ändern kann man das, indem der Boolean TAGGIT\_CASE\_INSENSITIVE in der settings.py auf True gestellt wird.
|
||||
\end{addmargin}
|
||||
Im ˜\texttt{model.py} wird das Feld \texttt{tag} neu erstellt und als Taggable Manager definiert.
|
||||
Au"serdem muss taggit in der \texttt{settings.py} unter \texttt{INSTALLED\_APPS} ergänzt werden. Um dem Programm mitzuteilen, dass nun eine neue Liste der Datenbank hinzugefügt werden muss, werden über die Kommandozeile Migrations-Befehle ausgeführt, die im Kapitel Datenmodellierung genauer beschrieben werden.
|
||||
Im Admin-Backend kann nun geprüft werden, ob das neue Feld in die Datenbank integriert wurde und neue Tags können in das Textfeld eingetragen werden. Der Parser verarbeitet jede Eingabe, die durch ein Komma oder ein Leerzeichen getrennt ist als einen Tag. Soll der Tag jedoch aus mehreren Wörtern bestehen, so müssen diese mit Anführungszeichen umfasst werden. Standardmä"sig unterscheidet der Taggable Manager zwischen Gro"s- und Kleinschreibung, Tags sind also case sensitive.
|
||||
|
||||
\subsection {Hilfsbibliotheken}
|
||||
Weitere Add-ons werden geladen, um kleinere Funktionen der Website einfach umsetzen zu können.
|
||||
|
||||
Zu diesen gehört \texttt{django-taggit-templatetags}, welches durch die Einbindung im HTML-File die Tags der Applikation als Liste ausgibt. Au"serdem lassen sich die eingepflegten Tags als Cloud visualisieren. Kommen bestimmte Schlagwörter öfter vor als andere, so werden sie entsprechend grö"ser dargestellt.
|
||||
|
||||
\texttt{Django-hitcount} dient zum zählen der Besucher einer Seite (vgl. [Tim15]). Es lässt sich auf drei verschiedene Arten in die Applikation einbinden. Der schnellste Weg ist die Darstellung der Besuche mit Hilfe eines Template Tags im HTML-File. Möchte man die Anzeige aber individueller gestalten, so kann durch das integrieren der \texttt{HitCountDetailView} in \texttt{views.py} die Variable \texttt{hitcount} verwenden und im Frontend ausgeben. Eine weitere Möglichkeit ist das Erweitern oder neu Erstellen eines Models im Django Backend. Von dort kann auf das neue Feld im Django-Admin-Backend zugegriffen werden, ebenso wie von der View und vom Template. Die im Add-on integrierten Einstellungen, die in der \texttt{settings.py} ergänzt werden müssen, ermöglichen unter anderem das Begrenzen der Lebensdauer des Zählers, bevor dieser zurück gesetzt wird.
|
||||
|
||||
Zum Versenden und Verwalten von E-Mails in Django eignet sich \texttt{django-post-office} (vgl. [Ong18]). Nach der Installation kann im Admin-Backend ein E-Mail-Template angelegt, Anhänge verwaltet und das Versenden dieser im Log überprüft werden. Es ist möglich die Benachrichtigungen mithilfe eines eines Shell-Skript des Frameworks \texttt{Cron}\footnote{ Cron dient zur zeitbasierten Ausführung von definierten Aufgaben.} asynchron zu versenden. Der Inhalt kann Text oder HTML-basiert sein und in mehreren Sprachen hinterlegt werden.
|
||||
|
||||
|
||||
\section{Bootstrap}
|
||||
Eine umfangreiche Website einheitlich zu gestalten ist oft sehr komplex und zeitaufwendig. Die Entwickler von Twitter haben deshalb, zunächst Firmenintern, an einem neuen Verwaltungswerkzeug gearbeitet, das mehrere Bibliotheken zusammenführen sollte. Sie merkten, das die neue Bibliothek, die daraus entstand, nicht nur auf Ihre eigene Website anwendbar, sondern so flexible ist, dass jede Art von Website davon profitieren könnte. 2011 entschieden Sie sich Bootstrap für die Öffentlichkeit frei zugänglich zu machen. Die Open-Source-Bibliothek, die auf GitHub abrufbar ist, wird seitdem von vielen interessierten Programmierern weiterentwickelt und ist somit stark gewachsen. Version 2.0 verfügt au"serdem über die Fähigkeit Websites responsive auf verschiedenste mobile Endgeräte anzupassen.
|
||||
Um die Usability des Prototyps zu erhöhen, wird das Framework Bootstrap eingebunden.
|
||||
|
||||
Das Bootstrap-Paket beinhaltet vorgefertigte Cascading Stylesheets, kurz CSS, die Farben, Schriftarten und viele weitere Stildefinitionen. Zudem befinden sich auch Erweiterungen des JavaScript-Frameworks jQuery in der Bibliothek, die weiterführende Funktionen beinhalten wie zum Beispiel Filter oder Dropdown-Menüs. Das Paket kann einfach eingebunden werden im head-tag einer HTML-Datei (vgl. Abbildung 2.3). Das bedeutet, dass Medie-Queries oder ähnliche Methoden nicht mehr nötig sind, nicht nur um eine Website mobilfähig zu machen, sondern auch kompatibel für die verschiedensten Browser.
|
||||
Eine umfangreiche Website einheitlich zu gestalten ist oft sehr komplex und zeitaufwendig. Die Entwickler von Twitter haben deshalb an einem neuen Verwaltungswerkzeug gearbeitet, das mehrere Bibliotheken zusammenführen sollte (vgl. [Ott11]). Die Open-Source-Bibliothek, die auf GitHub abrufbar ist, wird seitdem von vielen Programmierern weiterentwickelt und ist somit stark gewachsen. Ab Version 2.0 verfügt diese über die Fähigkeit, Websites \textit{responsiv}\footnote{ Responsive Webseiten sind auf allen Endgeräten angepasst darstellbar.} auf verschiedenste mobile Endgeräte anzupassen (vgl. [Ott12]).
|
||||
|
||||
Das Bootstrap-Paket beinhaltet vorgefertigte Cascading Stylesheets, kurz CSS, die Farben, Schriftarten und viele weitere Stildefinitionen implizieren. Zudem befinden sich auch Erweiterungen des JavaScript-Frameworks jQuery in der Bibliothek, die weiterführende Funktionen beinhalten wie zum Beispiel Filter oder Dropdown-Menüs. Das Paket kann im head-tag einer HTML-Datei (vgl. Abbildung 2.4.) einfach eingebunden werden. Das bedeutet, dass \textit{Media-Queries}\footnote{ Media-Qeries setzen statische Umbrüche um Layouts verschiedener Endgeräte anzupassen.} oder ähnliche Methoden nicht mehr nötig sind - nicht nur um eine Website mobilfähig zu machen, sondern auch kompatibel für die verschiedensten Browser (vgl. [Boo12]).
|
||||
|
||||
\begin{figure}[!h]
|
||||
\centering
|
||||
@ -127,7 +116,7 @@ Das Bootstrap-Paket beinhaltet vorgefertigte Cascading Stylesheets, kurz CSS, di
|
||||
\hfill
|
||||
\end{figure}
|
||||
|
||||
Durch das Einbinden von Bootstrap in einer HTML-Datei werden einige Styles bereits automatisch auf die darin vorkommenden Tags, wie Links und Überschriften, angewandt. Dies ist jedoch nur ein sehr kleiner Teil den die Bibliothek zur Verfügung stellt. Möchte man Bootstrap umfangreich nutzen so lassen sich die Stildefinitionen mit Klassen oder ID's in diverse HTML-Tags eintragen (vlg. Abbildung 2.4.).
|
||||
Durch das Einbinden von Bootstrap in einer HTML-Datei werden einige Styles bereits automatisch auf die darin vorkommenden Tags, wie Links und Überschriften, angewendet. Dies ist jedoch nur ein sehr kleiner Teil, den die Bibliothek zur Verfügung stellt. Bootstrap lässt sich umfangreicher nutzen, indem Stildefinitionen mit Klassen oder ID's in diverse HTML-Tags eingetragen werden (vlg. Abbildung 2.5.).
|
||||
|
||||
\begin{figure}[!h]
|
||||
\centering
|
||||
@ -136,9 +125,5 @@ Durch das Einbinden von Bootstrap in einer HTML-Datei werden einige Styles berei
|
||||
\hfill
|
||||
\end{figure}
|
||||
|
||||
Möchte man bestimmte gestalterische Eigenschaften von Bootstrap überschreiben muss eine eigens verfasste CSS-Datei nach der Verlinkung von Bootstrap in die Website eingebunden werden. Der Parser liest die Datei von oben nach unten, Links nach Rechts. Liest dieser also zu erst die Bootstrap Bibliothek und speichert diese, so überschreiben die Styles die danach kommen, die bereits gelesenen Eingaben. Die Styles, die inline auf ein Tag angewendet werden sind somit die bestimmenden Eigenschaften. Natürlich sollten Stildefinitionen niemals inline eingepflegt werden, da dies zu einem sehr unübersichtlichen und wartungsintensiven Code führen.
|
||||
|
||||
|
||||
|
||||
|
||||
Möchte man bestimmte gestalterische Eigenschaften von Bootstrap überschreiben, muss eine eigens verfasste CSS-Datei nach der Verlinkung von Bootstrap in die Website eingebunden werden. Der Parser liest die Datei von oben nach unten, links nach rechts. Liest dieser also zu erst die Bootstrap Bibliothek und speichert diese, so überschreiben die Styles, die danach kommen, die bereits gelesenen Eingaben. Die Styles, die inline auf ein Tag angewendet werden, sind somit die bestimmenden Eigenschaften.
|
||||
|
||||
|
@ -1,25 +1,203 @@
|
||||
\chapter{Prototyp}
|
||||
\label{ch:prototyp}
|
||||
Um zu beweisen, wie eine Spcial Media Plattform die Mail-Flut der Efi-Fakultät reduzieren kann, wird ein Prototyp implementiert.
|
||||
Um die Forschungsfrage zu prüfen, wird in dieser Arbeit die Methode des Prototypings genutzt. Der Prototyp dient zum experimentellen Arbeiten und sichert eine strukturell fundierte Umsetzung des Endprodukts. Der Fokus liegt dabei zunächst auf der Funktionalität der Anwendung. Prototyping wird als bevorzugte Methode gewählt, um schnell ein Ergebnis zu erzielen (vgl. [Abr16]). Zudem soll darauf aufbauend ein Produkt realisiert werden, das als Erweiterung in das Netzwerk der Hochschule integriert wird.
|
||||
|
||||
\section{Forschungsdesign}
|
||||
Dieses Kapitel veranschaulicht eine kurze Übersicht der Vorgehensweise und den Leitfaden, an den sich die Implementierung des Prototyps anlehnt (vgl. Abbildung 3.1.).
|
||||
Zu Beginn der Arbeit wird das sich aus der Forschungsfrage ergebenden Problem analysiert und es werden alle wichtigen Anforderungen erfasst. Dies bildet die Basis für alle weitern notwendigen Schritte, um am Ende eine sinnvolle Lösung bereitstellen zu können. Die Recherche dient der Sammlung aller notwendigen Werkzeuge und gibt einen Überblick über verschiedene Hilfsbibliotheken. Die Implementierung der Applikation kann nun auf Basis der Recherche durchgeführt werden. Dazu gehört das Testen verschiedener Bibliotheken und Erweiterungen, um das bestmögliche Ergebnis zu eruieren. Abschlie"send wird die Funktionalität des Prototypen getestet und evaluiert, ob die Forschungsfrage ausreichend beantwortet wird. Handlungsempfehlungen und mögliche Funktionen zum Erweitern finalisieren die Arbeit.
|
||||
|
||||
\begin{figure}[!h]
|
||||
\centering
|
||||
\includegraphics[width=0.8\textwidth]{figures/forschungsdesign}
|
||||
\caption{Forschungsdesign}
|
||||
\hfill
|
||||
\end{figure}
|
||||
|
||||
\section{Organisation}
|
||||
Grundlegender Aufbau der Website, Verwaltung der Daten evlt auf nochmal Taggable-Manager (ManyToMany) ...
|
||||
|
||||
\subsection{Verwaltung im Administrator-Backend}
|
||||
Näher auf CustomUserModel eingehen, Diagramm erstellen und einbinden
|
||||
|
||||
\subsection{Berechtigung der User}
|
||||
Welche Berechtigungen gibt es im Prototyp, welche werden vom Active Directory übernommen?
|
||||
Um einen Einblick in den Aufbau eines Django-Projektes zu erlangen wird die Struktur im Folgenden genauer beschrieben. Die unterste Projektebene wird durch \texttt{manage.py} gebildet. Sie wird unter anderem genutzt, um den lokalen Server starten zu können. In der Ebene darüber findet sich im Ordner \texttt{mysite} die Datei \texttt{settings.py}. Hier werden die allgemeinen Einstellungen der Website vorgenommen, wie zum Beispiel das Integrieren der Erweiterungen und der Pfad zu den hinterlegten Templates. Au"serdem ist die \texttt{urls.py} dort zu finden, deren Funktion bereits im Kapitel Django erläutert wurde. Im Ordner \texttt{thisisenv} sind alle Bibliotheken und Add-ons der virtuellen Umgebung hinterlegt. Der relevanteste Teil dieser Arbeit liegt im Ordner \texttt{application}. Hier sind die Datenbank-Migrationen, die Static-Files wie bootstrap und css, und alle Templates abgelegt. Zudem befindet sich hier die Logik des Prototypen, auf die im Kapitel \texttt{Funktionen} weiter eingegangen wird.
|
||||
|
||||
|
||||
\subsection{Datenmodellierung}
|
||||
Die Struktur der bereits bestehenden Datenbank im Django-Framework und die Erweiterungen desselben werden genauer erläutert. Zunächst wird auf die Ergänzung des bestehenden \texttt{UserModel} eingegangen, danach veranschaulicht der zweite Abschnitt das \texttt{PostModel} und abschlie"send werden die Zusammenhänge der Modelle dargestellt.
|
||||
|
||||
Alle Modelle werden als Django-Modelle deklariert um beim Kompilieren des Codes dem Compiler mitzuteilen, dass eine Integration stattfinden muss (vgl. [Dja18a]). Mit der Eingabe
|
||||
\\
|
||||
\begin{addmargin}[0pt]{0pt}
|
||||
\noindent\hspace*{10mm}%
|
||||
\texttt{ \$ python3 manage.py makemigrations}
|
||||
\\
|
||||
\end{addmargin}
|
||||
werden die neun Tabellen der Modelle erstellt. Zur letztendlichen Migration wird
|
||||
\\
|
||||
\begin{addmargin}[0pt]{0pt}
|
||||
\noindent\hspace*{10mm}%
|
||||
\texttt{ \$ python3 manage.py migrate}
|
||||
\\
|
||||
\end{addmargin}
|
||||
ausgeführt.\\
|
||||
|
||||
|
||||
\textbf{UserModel:}
|
||||
\begin{addmargin}[25pt]{0pt}
|
||||
Hier ist das Authentifizierungssystem von Django mit einem \texttt{UserModel} bereits angelegt. Dieses muss für den Prototyp um das Feld \texttt{tags} erweitert werden, sodass ein Benutzer folgende Felder besitzt (vgl. [Fou18a]):
|
||||
\begin{itemize}
|
||||
\item username, fist\_name, last\_name, email, groups, user\_permissions, is\_staff, is\_active, is\_superuser, last\_login, date\_joined, tags
|
||||
\end{itemize}
|
||||
|
||||
In models.py ist der \texttt{CustomUser} dafür verantwortlich, das neue Feld mit dem \texttt{Default-User} zu verknüpfen. Durch das \texttt{OneToOneField} (siehe Abbildung 3.2.) wird die Verbindung zum schon bestehenden Modell hergestellt. \texttt{OneToOne} bildet eine einzigartige Zuordnung von zwei Objekten, sodass der Rückgabewert eindeutig ist. Das hei"st, dass hier keine rekursiven, auf sich selbst verlinkende oder \texttt{lazy} Beziehungen möglich sind, um Konflikte bei der Authentifizierung zu vermeiden. Dies ist die übliche Vorgehensweise, um mit einem Primärschlüssel das Default-Model zu erweitern (vgl. [Fou18a]). \\
|
||||
|
||||
\begin{figure}[!h]
|
||||
\centering
|
||||
\includegraphics[width=1\textwidth]{figures/custommodelcode}
|
||||
\caption{CustomUserModel in models.py}
|
||||
\hfill
|
||||
\end{figure}
|
||||
|
||||
|
||||
\end{addmargin}
|
||||
|
||||
\textbf{PostModel:}
|
||||
\begin{addmargin}[25pt]{0pt}
|
||||
Das \texttt{PostModel} beschreibt alle Felder, die ein Artikel enthalten kann. Basierend auf der Blog-Lösung von Djangogirls.com (vgl. [Dja18b]) gehören dazu folgende:
|
||||
\begin{itemize}
|
||||
\item author, title, text, created\_date, published\_date, tags
|
||||
\end{itemize}
|
||||
Der Autor ist durch einen \texttt{ForeignKey} mit dem \texttt{UserModel} verbunden. Diese \texttt{ManyToOne} Verbindung reicht hier aus, um einem Post einem Autor, also dem eingeloggten User, zuzuweisen. \texttt{Title} ist ein \texttt{CharField} und wird mit einer Zeichenbegrenzung festgelegt. Der Text hingegen kann eine beliebige Menge an Zeichen enthalten und wird deshalb als \texttt{TextField} deklariert. Erstellungsdatum und Publikation sind \texttt{DateTimeFields}. Ersteres muss vom Ersteller angegeben werden, zweiteres kann zunächst durch die Zusatzangabe \texttt{null=True} offen gelassen werden. Ein weiteres Feld wird hinzugefügt, um Artikeln unabhängig von Usern Tags zuordnen zu können.
|
||||
\\
|
||||
\end{addmargin}
|
||||
|
||||
|
||||
\textbf{Gesamtmodellierung:}
|
||||
|
||||
\begin{addmargin}[25pt]{0pt}
|
||||
Die Abbildung 3.3. zeigt die Modellierung der Tabelle \texttt{User} und \texttt{Post}. Au"serdem verdeutlicht es die Erweiterung des User-Modells von Django mit dem in der Applikation angelegten \texttt{CustomUser}. Die im User vorkommenden \textit{booleschen Felder} werden im Kapitel \glqq Berechtigungen der User" genauer erörtert.
|
||||
\end{addmargin}
|
||||
|
||||
\begin{figure}[!h]
|
||||
\centering
|
||||
\includegraphics[width=0.9\textwidth]{figures/datamodel}
|
||||
\caption{Datenmodellierung von \texttt{User} und \texttt{Post}}
|
||||
\hfill
|
||||
\end{figure}
|
||||
|
||||
\subsection{Berechtigungen der User}
|
||||
Im Allgemeinen verwendet man Berechtigungen, um Benutzern Zugang zu bestimmten Resourcen in einem Netzwerk einzuräumen. Au"serdem bestimmt eine Staffelung die Art des Zugangs, ob der User die Resourcen nur lesen, verändern oder löschen darf (vgl. [Com18]). Die Rechte werden meist einzelnen Individuen oder einer Gruppe zugeordnet.
|
||||
|
||||
Das gestaffelte Berechtigungsmanagement ist im Prototyp notwendig, um den Umgang mit Informationen so sicher wie möglich zu gestalten und um die Nachhaltigkeit dieser zu bewahren. Des Weiteren soll der Prototyp als Vorlage für die Erweiterung der Hochschulwebsite dienen und daher ist eine, zur Hochschule ähnliche Verteilung der Zugangsberechtigungen sinnvoll.
|
||||
|
||||
Studenten sollen zunächst Informationen weder einpflegen noch editieren dürfen. Die einzigen Änderungen, die sie vornehmen können, sind auf ihre eigene Datenbank fokussiert. Dazu gehört das Hinzufügen von Tags, durch die damit verlinkten Beiträge das persönliche Dashboard befüllt wird. Dies soll verhindern, dass Informationen zu leichtfertig geändert oder gelöscht werden.
|
||||
|
||||
Dozenten und Angestellte der Hochschule sind dazu berechtigt, Posts zu erstellen, zu editieren und wieder zu löschen. Zudem können sie, wie Studenten, Tags abonnieren und das persönliche Dashboard gestalten. Das Einloggen in die Administratoroberfläche kann durchgeführt werden, jedoch sind der Gruppe noch keinerlei Rechte zugewiesen. Möchte man das ändern, kann man das von Django bereitgestellte Feld \texttt{User Permissions} im Admin-Backend unter \texttt{Users} finden, und dem Namen der Person die gewünschte Berechtigung zuteilen. Diese sind von Django vorgegeben und betreffen alle vorhandenen Modelle der Applikation.
|
||||
Durch das Setzen des booleschen Wert \texttt{is\_staff} auf \texttt{True} beim Erstellen der Benutzer ist es möglicha im Code der Applikation Abfragen durchzuführen. Dadurch lassen sich bestimmte Views an die eingeloggte Personengruppe anpassen. So ist unter Anderem das Menü der Dozenten und Angestellten etwas umfangreicherer als das der Studierenden (vgl. Abbildung 3.4.).\\
|
||||
|
||||
\begin{figure}[!h]
|
||||
\centering
|
||||
\includegraphics[width=1\textwidth]{figures/menuadmin}
|
||||
\caption{Menü für eingeloggte Benutzer mit Adminrechte}
|
||||
\hfill
|
||||
\end{figure}
|
||||
|
||||
\section{Funktionen}
|
||||
Um die wichtigsten Funktionen des Prototypen festlegen zu können, werden User Stories erstellt (vgl. Abbildung 3.5.). Diese bestehen aus kurzen Sätzen und beschreiben aus Sicht des Nutzers das Verwenden einer Funktion. Die Priorisierung bezieht sich hierbei auf die Relevanz der Funktion, wobei die Funktionen mit einem rotem Punkt sehr wichtig für den Prototypen sind, Orangene wichtige Funktionen sind, aber nicht unbedingt notwendig, und Grüne kaum Relevanz haben. \\
|
||||
|
||||
\begin{figure}[!h]
|
||||
\centering
|
||||
\includegraphics[width=0.9\textwidth]{figures/userstories}
|
||||
\caption{User Stories}
|
||||
\hfill
|
||||
\end{figure}
|
||||
|
||||
\subsection{Verwaltung der Funktionen}
|
||||
Das Verwalten der Artikel soll von berechtigten Nutzern im Frontend stattfinden, während die prozessuale Logik im Code-Backend realisiert ist. Der Vorgang des Erstellens, des Löschens und des Editierens der einzelnen Einträge wird im Folgenden konkretisiert.
|
||||
\\
|
||||
\\
|
||||
\textbf{Einen neuen Artikel erstellen: }
|
||||
|
||||
Das \texttt{+} in der Menüleiste leitet den Benutzer zu einer Unterseite. Hier können alle Felder befüllt werden, die im \texttt{PostForm}-Formular in der Datei \texttt{forms.py} festgelegt wurden. Dazu gehören der Titel und der Text, die als Pflichtfelder gelten. Das Feld \texttt{Tags} muss ebenfalls mindestens einen Wert enthalten, um die Validierung der Eingaben sichern zu können. Eine Ausnahme bildet das Datum der Veröffentlichung. Bleibt das Feld leer, so wird der Beitrag automatisch der Liste der Entwürfe beigefügt.
|
||||
|
||||
Speichert der Benutzer den Artikel, so werden im Backend die Daten wie folgt verarbeitet. In der View \texttt{post\_new} wird zunächst geprüft, ob alle Eingaben valide sind. Falls dies der Fall ist, wird der jeweilige Beitrag als Objekt zurückgegeben, jedoch durch das optionale Keywort \texttt{commit=false} noch nicht in der Datenbank gespeichert. Das ist notwendig, um dem Objekt spezifische Informationen mitzugeben. In diesem Kontext wird der aktuell eingeloggte User als Autor hinterlegt. Jedoch birgt die Vorgehensweise eine Problematik im Speichervorgang einer \texttt{ManyToMany} Relation zwischen zwei Modellen. Da Informationen nur auf ein bereits in der Datenbank bestehendes Objekt gesichert werden können, ist dies zunächst nicht möglich (vgl. [Fou18b]).
|
||||
Im Prototyp nutzt das \texttt{PostModel} die \texttt{ManyToMany} Konnektivität mit dem Modell des \texttt{TaggabelManagers}. Um die Eingabe des Tag-Felds trotzdem im neuen Artikel speichern zu können, wird zunächst das Objekt gespeichert, um danach explizit das von Django zur Verfügung gestellte \texttt{form.save\_m2m()} aufrufen zu können. Der Befehl zwingt die Daten der \texttt{ManyToMany} Relation zu speichern.
|
||||
|
||||
Die eindeutige Zuordnung der Eingabe im Frontend zur Verarbeitung der Artikel im Backend ist mit einem \texttt{Primary Key} realisiert. Das \texttt{PostModel} bekommt beim Anlegen keinen Schlüssel zu einem Feld zugewiesen. Django erstellt ihn automatisiert beim Speichern der Tabelle als \texttt{AutoField} im Feld \texttt{Id} und identifiziert die \texttt{Id} automatisch bei jedem neu Erstellen eines Objekts. Somit sind alle Objekte eindeutig zuordenbar und können mit dem Kommando \texttt{post.pk} jederzeit abgefragt werden.
|
||||
\\
|
||||
\\
|
||||
\textbf{Einen bereits vorhandenen Artikel löschen: }
|
||||
|
||||
In der Detailansicht eines Artikels ist es möglich, denselben zu entfernen. Die View \texttt{post\_remove} selektiert über den im Template mitgegebenen \texttt{Primary Key} das Objekt und speichert es in der Variable \texttt{post}. Der Post wird gelöscht mit dem Befehl \texttt{post.remove()} und eine Umleitung am Ende der View-Definition schickt den Benutzer auf die Seite der Artikelliste. Hier wird eine zuvor in der View generierte Nachricht visualisiert, sodass der Benutzer sicher sein kann, dass der Vorgang abgeschlossen ist.
|
||||
\\
|
||||
\\
|
||||
\textbf{Einen bereits vorhandenen Artikel bearbeiten: }
|
||||
|
||||
Ähnlich wie beim Löschen eines Artikel, kann man diesen in der Detailansicht bearbeiten. Dazu wird in der View über den \texttt{Primary Key} der Artikel einer Variable \texttt{post} zugeordnet. Die bedingte Anweisung rendert zunächst die \texttt{PostForm}, mit dem bereits eingepflegten Inhalt durch eine GET-Abfrage (vgl. Abbildung 3.6.).
|
||||
|
||||
\begin{figure}[!h]
|
||||
\centering
|
||||
\includegraphics[width=0.9\textwidth]{figures/postedit}
|
||||
\caption{Prototyp Artikel-Editor.}
|
||||
\hfill
|
||||
\end{figure}
|
||||
|
||||
Veranlasst der Benutzer die Speicherung des Artikels im Frontend, wird die bedingte Abfrage der Abbildung 3.7. in Zeile 91 erfüllt. Die POST-Abfrage ist hier notwendig, da Django nur so Daten in der Datenbank verändert. Eine Begründung hierfür ist die Art der Übertragung der Daten an den Server. \texttt{POST-Requests} bündeln alle Daten, verschlüsseln sie und senden sie dann an der Server (vgl. [Fou18c]). Dadurch ist der Vorgang einfacher kontrollierbar und mit einem \texttt{csrf-Token} im Template ebenfalls gegen Cross-Site-Request-Fälschung abgesichert. Die weitere Vorgehensweise der Funktion ist identisch zum bereits dargelegten neu erstellen eines Artikels und muss nicht weiter beschrieben werden.\\
|
||||
|
||||
\begin{figure}[!h]
|
||||
\centering
|
||||
\includegraphics[width=0.9\textwidth]{figures/post-edit-view}
|
||||
\caption{Funktion post\_edit, Auszug aus views.py.}
|
||||
\hfill
|
||||
\end{figure}
|
||||
|
||||
\subsection{Artikel abonnieren}
|
||||
|
||||
Das Abonnieren bestimmter Themengebiete ist eine der wichtigsten Funktionen im Prototyp, um die eingepflegten Informationen zielgerichtet anzeigen zu können.
|
||||
|
||||
Unter Berücksichtigung aller Vor- und Nachteile wird ein Tag-Modell zur Umsetzung gewählt. Wie bereits in der Datenmodellierung angedeutet (vgl. Datenmodellierung), besitzt jeder Artikel beschreibende Tags. Hierbei handelt es sich um kurze stichwortartige Beschreibungen, die den Artikel so gut wie möglich charakterisieren. Abhängig vom Umfang des Blogsystems, sollte die Anzahl der Tags immer in einem gewissen Rahmen vorhanden sein. Das bedeutet zum einen, dass Ersteller von Artikeln immer die gleiche Menge an Schlagwörtern verwenden, wobei geringe Abweichungen möglich sind (vgl. [Gmb18]). Hat das System bereits einen grö"seren Umfang angenommen, sollten zum anderen keine neuen Tags erstellt werden, um die Übersicht für Autoren und Leser zu bewahren. \\
|
||||
|
||||
\begin{figure}[!h]
|
||||
\centering
|
||||
\includegraphics[width=1\textwidth]{figures/filtern}
|
||||
\caption{Prototyp Suche- und Abonnier-Seite}
|
||||
\hfill
|
||||
\end{figure}
|
||||
|
||||
Im Prototyp findet man die Abonnement-Funktion unter dem Menüpunkt \texttt{Suche}. Hier erscheint ein zweigeteiltes Layout, welches auf der rechten Seite alle bereits abonnierten Tags auflistet und darunter die Eingabe eines neuen Tags ermöglicht. Um den Benutzer alle bereits existierenden Tags offen zu legen, befindet sich auf der linken Seite des Layouts eine \textit{Tag-Cloud} \footnote{ Tag-Cloud ist eine Visualisierung eines Schlagwortverzeichnisses.}
|
||||
, die alle Schlagwörter darstellt (vgl. Abbildung 3.8.).
|
||||
|
||||
|
||||
Die Eingabe des zu abonnierenden Tags wird durch ein Formular realisiert. Dieses ist in der \texttt{forms.py} Datei konfiguriert und enthält nur ein Eingabefeld. Der Ablauf verläuft gleichartig zum oben dargestellten Erstellen eines Artikels, wird allerdings genauer beschrieben, um die Struktur des \texttt{Taggable Managers} zu verdeutlichen.
|
||||
|
||||
\begin{figure}[!h]
|
||||
\centering
|
||||
\includegraphics[width=1.1\textwidth]{figures/view-search-add}
|
||||
\caption{Funktion search\_add, Auszug aus views.py.}
|
||||
\hfill
|
||||
\end{figure}
|
||||
|
||||
Gibt der Benutzer einen Tag ein und sendet durch Betätigen des Sichern-Buttons den \texttt{Request}, wird dieser in der \texttt{views.py} verarbeitet. In Zeile 159 der Abbildung 3.9. wird der eingeloggte Benutzer der Variable \texttt{user\_instance} übergeben. Beim Erstellen der \texttt{Model-Instanz} (vgl. Abbildung 3.9., Zeile 161) wird \texttt{user\_instance} der Unbekannten \texttt{form} zugeteilt, um die Tag-Eingabe im richtigen User-Objekt integrieren zu können. Nach der Abfrage der Formvalidität wird ein neues Objekt angelegt (vgl. Abbildung 3.9., Zeile 163) und ebenfalls dem aktuellen Benutzer zugeordnet. Die Eingabe der \texttt{form} wird in einem \texttt{Array} zwischengespeichert und mit dem Attribut \texttt{cleaned\_data} in ein für Python kompatiblen Datentyp gecastet. Um prüfen zu können, ob die Eingaben der Form tatsächlich im \texttt{Tag-Model} enthalten sind, wird diese nochmals in einen String umgewandelt und mit den bereits existierenden Tags abgeglichen (vgl. Abbildung 3.9., Zeile 168). Wird die Bedingung erfüllt, speichert die Funktion die Tags. In beiden möglichen Fällen wird der Benutzer benachrichtigt, ob der Vorgang erfolgreich oder die Eingabe nicht valide war.
|
||||
|
||||
Nun werden auf dem Dashboard Artikel der neu hinzugefügten Tags angezeigt (vgl Abbildung 3.10.).\\
|
||||
|
||||
\begin{figure}[!h]
|
||||
\centering
|
||||
\includegraphics[width=1\textwidth]{figures/newsfeed}
|
||||
\caption{Prototyp Newsfeed Seite}
|
||||
\hfill
|
||||
\end{figure}
|
||||
|
||||
|
||||
\subsection{Filtern von Artikeln}
|
||||
|
||||
Zur Unterstützung der Nutzbarkeit des Prototypen ist es wichtig, dass User intuitiv nach Tags suchen und sie selektieren können. Hierfür werden verschiedene Möglichkeiten zur Verfügung gestellt, die die Usability der Website verbessern sollen.
|
||||
|
||||
Im persönlichen Newsfeed des Dashboards sind die zu den Artikeln zugewiesenen Schlagwörter jeweils mit Verlinkungen versehen. Möchte ein Benutzer weitere Artikel zu einem bestimmten Thema lesen, so muss er lediglich auf den entsprechenden Tag klicken und erhält eine Liste aller Beiträge, die den selektierten Tag enthalten. Hierfür wird keine eigene \texttt{View} benötigt, denn das Erstellen von Listen mit unterschiedlichem Inhalt kann ebenso über sich unterscheidende Urls realisiert werden. Im Template \texttt{post\_list} wird beim Klicken auf einen Tag der \textit{Slug}\footnote{ Der Slug dient im Taggable-Manager als eindeutig zuweisbarer Identifikator.} desselben mitgegeben. Au"serdem wird nun die Url \texttt{post\_list\_by\_tag} aufgerufen, die auf eine neue Seite verweist. Die View \texttt{post\_list} rendert bei Anfragen mit \textit{Slug} die passende Liste, bei Anfragen ohne werden alle vorhandenen Artikel geladen.
|
||||
|
||||
|
||||
Der Prototyp bietet zudem die Möglichkeit direkt nach Tags zu suchen. Unter dem Menüpunkt \texttt{Suche} auf der linken Seite des geteilten Layouts befindet sich eine \textit{Tag-Cloud}. Darunter ist das Suchfeld, welches durch ein Formular im Template realisiert wird. Die Eingabe fragt jedoch nur Tags aus der Datenbank ab, die Daten werden nicht verändert, da hier lediglich ein \texttt{GET-Request} gesendet wird. Mit der Funktion \texttt{filter} und dem von der taggit-Bibliothek zur Verfügung gestellten \textit{Lookup} \footnote{ Lookup ist eine Art Funktion, die den Ort des darauffolgend Feldes ausgibt.}
|
||||
\texttt{tags\_\_name\_\_in} können alle Artikel mit dem jeweils enthaltenen Tag dem Template übergeben werden.
|
||||
|
||||
|
||||
|
||||
|
||||
\section{Funktion}
|
||||
Nötige Funktionen
|
||||
|
||||
\subsection{Abonnieren}
|
||||
Tags als eingeloggter User abonnieren und verwalten Front-end und Admin-Backend?
|
||||
|
||||
\subsection{Filtern}
|
||||
Tag-map? Filtern nach abonnierten Posts, alle Posts und Posts mit bestimmten Tags
|
||||
|
||||
\subsection{Benachrichtigung}
|
||||
Mail-Benachrichtigung wöchentlich
|
||||
|
BIN
doc/bachelorarbeit_EstherKleinhenz/figures/crontab.png
Normal file
After Width: | Height: | Size: 18 KiB |
BIN
doc/bachelorarbeit_EstherKleinhenz/figures/custommodelcode.png
Normal file
After Width: | Height: | Size: 31 KiB |
BIN
doc/bachelorarbeit_EstherKleinhenz/figures/datamodel.pages
Executable file
BIN
doc/bachelorarbeit_EstherKleinhenz/figures/datamodel.png
Normal file
After Width: | Height: | Size: 192 KiB |
After Width: | Height: | Size: 52 KiB |
BIN
doc/bachelorarbeit_EstherKleinhenz/figures/evaluation-1.png
Normal file
After Width: | Height: | Size: 84 KiB |
BIN
doc/bachelorarbeit_EstherKleinhenz/figures/evaluation-2.png
Normal file
After Width: | Height: | Size: 105 KiB |
BIN
doc/bachelorarbeit_EstherKleinhenz/figures/evaluation-3.png
Normal file
After Width: | Height: | Size: 111 KiB |
BIN
doc/bachelorarbeit_EstherKleinhenz/figures/evaluation-4.png
Normal file
After Width: | Height: | Size: 29 KiB |
BIN
doc/bachelorarbeit_EstherKleinhenz/figures/evaluation-5.png
Normal file
After Width: | Height: | Size: 31 KiB |
BIN
doc/bachelorarbeit_EstherKleinhenz/figures/evaluation.numbers
Executable file
BIN
doc/bachelorarbeit_EstherKleinhenz/figures/filtern.png
Normal file
After Width: | Height: | Size: 96 KiB |
BIN
doc/bachelorarbeit_EstherKleinhenz/figures/forschungsdesign.png
Normal file
After Width: | Height: | Size: 78 KiB |
BIN
doc/bachelorarbeit_EstherKleinhenz/figures/menuadmin.png
Normal file
After Width: | Height: | Size: 30 KiB |
BIN
doc/bachelorarbeit_EstherKleinhenz/figures/newsfeed.png
Normal file
After Width: | Height: | Size: 174 KiB |
BIN
doc/bachelorarbeit_EstherKleinhenz/figures/post-edit-view.png
Normal file
After Width: | Height: | Size: 118 KiB |
BIN
doc/bachelorarbeit_EstherKleinhenz/figures/postedit.png
Normal file
After Width: | Height: | Size: 85 KiB |
BIN
doc/bachelorarbeit_EstherKleinhenz/figures/userstories.png
Normal file
After Width: | Height: | Size: 153 KiB |
BIN
doc/bachelorarbeit_EstherKleinhenz/figures/userstory.pages
Executable file
BIN
doc/bachelorarbeit_EstherKleinhenz/figures/view-search-add.png
Normal file
After Width: | Height: | Size: 262 KiB |
@ -1,6 +1,6 @@
|
||||
%------------------------------------------------------------------------------
|
||||
%geometry
|
||||
\usepackage[a4paper,inner=2cm,outer=4cm,top=3cm,bottom=3cm]{geometry}
|
||||
\usepackage[a4paper,inner=4cm,outer=2cm,top=3cm,bottom=3cm]{geometry}
|
||||
\renewcommand{\topfraction}{.85}
|
||||
\renewcommand{\bottomfraction}{.85}
|
||||
%load packages-----------------------------------------------------------------
|
||||
@ -14,6 +14,8 @@
|
||||
\usepackage{float}
|
||||
\usepackage[onehalfspacing]{setspace}
|
||||
\usepackage{textcomp}
|
||||
\usepackage{hyperref}
|
||||
\hypersetup{colorlinks=true,linkcolor=black}
|
||||
%Abkürzungsverzeichnis
|
||||
\usepackage[printonlyused]{acronym}
|
||||
|
||||
@ -42,6 +44,7 @@
|
||||
\usepackage[nottoc]{tocbibind}
|
||||
%------------------------------------------------------------------------------
|
||||
\usepackage{titlesec}
|
||||
\usepackage{pdfpages}
|
||||
%------------------------------------------------------------------------------
|
||||
\usepackage{comment}
|
||||
%------------------------------------------------------------------------------
|
||||
|
@ -47,9 +47,159 @@
|
||||
note = "https://docs.python.org/3/tutorial/venv.html"
|
||||
}
|
||||
|
||||
@article{Dixit,
|
||||
author = "Chitrank Dixit",
|
||||
title = "PEP-8 Tutorial: Code Standards in Python",
|
||||
year = "2018",
|
||||
note = "https://www.datacamp.com/community/tutorials/pep8-tutorial-python-code"
|
||||
@article{Shabda,
|
||||
author = "Shabda",
|
||||
title = "Understanding decorators",
|
||||
year = "2009",
|
||||
note = "https://www.agiliq.com/blog/2009/06/understanding-decorators/"
|
||||
}
|
||||
|
||||
@article{Timm,
|
||||
author = "Damon Timm",
|
||||
title = "django-hitcount Documentation",
|
||||
year = "2015",
|
||||
note = "https://django-hitcount.readthedocs.io/en/latest/overview.html"
|
||||
}
|
||||
|
||||
@article{Ong,
|
||||
author = "Selwin Ong",
|
||||
title = "Django Request-Response Cycle",
|
||||
year = "2018",
|
||||
note = "https://github.com/ui/django-post\_office/blob/master/AUTHORS.rst"
|
||||
}
|
||||
|
||||
@article{Nevius,
|
||||
author = "Ryan Nevius",
|
||||
title = "django-post\_office git Repository",
|
||||
year = "2015",
|
||||
note = "https://ryannevius.com"
|
||||
}
|
||||
|
||||
@article{Solutions,
|
||||
author = "Mindfire Solutions",
|
||||
title = "Advantages and Disadvantages of Python Programming Language",
|
||||
year = "2017",
|
||||
note = "https://medium.com/@mindfiresolutions.usa/advantages-and-disadvantages-of-python-programming-language-fd0b394f2121"
|
||||
}
|
||||
|
||||
@book{Weigend,
|
||||
author = "Michael Weigend",
|
||||
title = "Python GE-PACKT",
|
||||
year = "2017",
|
||||
note = "Kapitel 23.1"
|
||||
}
|
||||
|
||||
@article{Otto,
|
||||
author = "Mark Otto",
|
||||
title = "Bootstrap from Twitter",
|
||||
year = "2011",
|
||||
note = "https://blog.twitter.com/developer/en\_us/a/2011/bootstrap-twitter.html"
|
||||
}
|
||||
|
||||
@article{Bootstrap,
|
||||
author = "Bootstrap Authors, Twitter Inc",
|
||||
title = "Bootstrap Repository",
|
||||
year = "2011",
|
||||
note = "https://github.com/twbs/bootstrap"
|
||||
}
|
||||
|
||||
@article{Otto-2,
|
||||
author = "Mark Otto",
|
||||
title = "Say hello to Bootstrap 2.0",
|
||||
year = "2012",
|
||||
note = "https://web.archive.org/web/20120203191214/https://dev.twitter.com/blog/say-hello-to-bootstrap-2"
|
||||
}
|
||||
|
||||
@article{Django,
|
||||
author = "Django Software Foundation",
|
||||
title = "django.contrib.auth, User Model",
|
||||
year = "2018",
|
||||
note = "https://docs.djangoproject.com/en/2.1/ref/contrib/auth/"
|
||||
}
|
||||
|
||||
@article{Djangogirls,
|
||||
author = "Djangogirls",
|
||||
title = "Creating a blog post model",
|
||||
year = "2018",
|
||||
note = "https://tutorial.djangogirls.org/en/django\_models/"
|
||||
}
|
||||
|
||||
@article{Djangogirls-2,
|
||||
author = "Djangogirls",
|
||||
title = "We inspire women to fall in love with programming.",
|
||||
year = "2018",
|
||||
note = "https://djangogirls.org/"
|
||||
}
|
||||
|
||||
@article{Computer,
|
||||
author = "The Computer Language Company",
|
||||
title = "Definition of: user permissions",
|
||||
year = "2018",
|
||||
note = "https://www.pcmag.com/encyclopedia/term/58231/user-permissions"
|
||||
}
|
||||
|
||||
@article{Django-2,
|
||||
author = "Django Software Foundation",
|
||||
title = "Modelforms - The save() methode",
|
||||
year = "2018",
|
||||
note = "https://docs.djangoproject.com/en/dev/topics/forms/modelforms/\#the-save-method"
|
||||
}
|
||||
|
||||
@article{Django-3,
|
||||
author = "Django Software Foundation",
|
||||
title = "Working with Forms",
|
||||
year = "2018",
|
||||
note = "https://docs.djangoproject.com/en/dev/topics/forms/\#using-a-form-in-a-view"
|
||||
}
|
||||
|
||||
@article{Fiok,
|
||||
author = "Marzena Fiok",
|
||||
title = "E-Mail-Flut sorgt für Kostenlawine",
|
||||
year = "2014",
|
||||
note = "https://www.tecchannel.de/a/e-mail-flut-sorgt-fuer-kostenlawine,402338,3"
|
||||
}
|
||||
|
||||
@article{Django-4,
|
||||
author = "Django Software Foundation",
|
||||
title = "Cross Site Request Forgery protection",
|
||||
year = "2018",
|
||||
note = "https://docs.djangoproject.com/en/dev/ref/csrf/"
|
||||
}
|
||||
|
||||
@article{Sario,
|
||||
author = "Sario Marketing GmbH",
|
||||
title = "Tagging",
|
||||
year = "2018",
|
||||
note = "https://www.textbroker.de/tagging"
|
||||
}
|
||||
|
||||
@article{Djangogirls-2,
|
||||
author = "Djangogirls",
|
||||
title = "Was passiert, wenn jemand eine Webseite beim Server anfordert?",
|
||||
year = "2018",
|
||||
note = "https://tutorial.djangogirls.org/de/django/"
|
||||
}
|
||||
|
||||
@article{Moodle,
|
||||
author = "Moodle Dokumentation",
|
||||
title = "Aufbau einer Moodle-Site",
|
||||
year = "2015",
|
||||
note = "https://docs.moodle.org/35/de/Aufbau\_einer\_Moodle-Site"
|
||||
}
|
||||
|
||||
@article{Moodle-2,
|
||||
author = "Moodle Dokumentation",
|
||||
title = "Cron-Job",
|
||||
year = "2018",
|
||||
note = "https://docs.moodle.org/35/de/Cron-Job"
|
||||
}
|
||||
|
||||
|
||||
@article{Hellmuth,
|
||||
author = "Dirk Hellmuth",
|
||||
title = "Wenn E-Mails zur Last werden: Wege aus der Mail-Überflutung",
|
||||
year = "2012",
|
||||
note = "https://www.unternehmer.de/management-people-skills/138949-wenn-e-mails-zur-last-werden-wege-aus-der-mail-ueberflutung"
|
||||
}
|
||||
|
||||
|
||||
|
@ -2,85 +2,35 @@
|
||||
|
||||
%------------------------------------------------------------------------------
|
||||
\begin{center}
|
||||
\begin{figure}[!h]
|
||||
\includegraphics[width=0.7\textwidth]{figures/Ohm_logo}
|
||||
\hfill
|
||||
\end{figure}
|
||||
\vspace{12mm}
|
||||
\textbf{\large Technische Hochschule Nürnberg Georg Simon Ohm}\\
|
||||
\thispagestyle{empty}
|
||||
\setlength{\unitlength}{15cm}
|
||||
\line(1,0){1}\\
|
||||
\vspace{0.4cm}
|
||||
{\LARGE\textbf{\LARGE Entwicklung einer internen Social Media Plattform mit personalisierbarem Dashboard für Studierende}}
|
||||
\setlength{\unitlength}{15cm}
|
||||
\line(1,0){1}\\
|
||||
\textbf{\large Fakultät Elektrotechnik Feinwerktechnik Informationstechnik}\\
|
||||
\vspace{1.5cm}
|
||||
\textbf{\large Studiengang Media Engineering}\\
|
||||
\vspace{1.5cm}
|
||||
\textbf{\large Bachelorarbeit von}\\
|
||||
\textbf{\large Esther Beate Kleinhenz}\\
|
||||
|
||||
\vspace{2.0cm}
|
||||
\textbf{\huge{Esther Kleinhenz}}\\
|
||||
\vspace{1.2cm}
|
||||
{\large\textbf{ Entwicklung einer internen Social Media Plattform mit personalisierbarem Dashboard f{\"u}r Studierende}}
|
||||
\setlength{\unitlength}{15cm}
|
||||
|
||||
\vspace{0.8cm}
|
||||
{\textbf{WS 2018/2019}}\\
|
||||
\vspace{2.5cm}
|
||||
|
||||
|
||||
\vspace{3.2cm}
|
||||
|
||||
\raggedright{ Abgabedatum: 11. Dezember 2018}\\
|
||||
\raggedright { Betreuer: Prof. Dr. Oliver Hofmann}\\
|
||||
\vspace{0.4cm}
|
||||
{\Large \textbf{Matrikelnummer: }2649270}\\
|
||||
\vspace{2.5cm}
|
||||
{\LARGE \textbf{Bachelorarbeit}}\\
|
||||
%\vspace{0.3cm}
|
||||
{zur Erlangung des akademischen Grades}\\
|
||||
\vspace{0.5cm}
|
||||
\textbf{\Large Bachelor of Engineering}\\
|
||||
{\large Media Engineering}\\
|
||||
\vspace{2.5cm}
|
||||
{\Large \textbf{Erstprüfer: }Prof. Dr. Oliver Hofmann}\\
|
||||
{\Large \textbf{Zweitprüfer: }Prof. Dr. Matthias Hopf}\\
|
||||
%\vspace{2.0cm}
|
||||
%{Co-Advisor:}\\
|
||||
%{Name}\\
|
||||
%\vspace{2.0cm}
|
||||
%{Laborat\'orio de Vibra\c{c}\~oes de Sistemas Mec%\^anicos}\\
|
||||
%{Departamento de Engenharia Mec\^anica }\\
|
||||
%{Faculdade de Engenharia da Universidade do Porto}\\
|
||||
%\vspace{1.2cm}
|
||||
|
||||
\vspace{\fill}
|
||||
Nürnberg, {\today}\\
|
||||
\noindent { Schlagworte: Python, Django, Informationsplattform, Tagging, E-Mail-Flut}
|
||||
\end{center}
|
||||
|
||||
\newpage
|
||||
\thispagestyle{empty}
|
||||
|
||||
\setlength{\unitlength}{15cm}
|
||||
\line(1,0){1}\\
|
||||
%\indent Diese Dissertation wurde\\
|
||||
%\indent Studiengang Wirtschaftsinformatik\\
|
||||
%\indent Fakultät Informatik\\
|
||||
%\indent Technischen Hochschule Nürnberg\\
|
||||
%\indent Georg Simon Ohm \\
|
||||
%\indent Nürnberg, Deutschland\\
|
||||
%\indent verfasst.
|
||||
%\vspace{1.5cm}
|
||||
|
||||
\indent Hiermit erkläre ich, dass die vorliegende Arbeit von mir selbständig verfasst und nicht\\
|
||||
\indent anderweitig für Prüfungszwecke vorgelegt wurde, keine anderen als die angegebenen\\
|
||||
\indent Quellen oder Hilfsmittel benutzt sowie wörtliche oder sinngemäße Zitate als solche\\
|
||||
\indent gekennzeichnet wurden.
|
||||
|
||||
\vspace{0.5cm}
|
||||
|
||||
\begin{flushright}
|
||||
Nürnberg, \today
|
||||
\end{flushright}
|
||||
|
||||
\vspace{1.0cm}
|
||||
|
||||
\indent Katja Cornelia Hader \\
|
||||
\indent E-mail: {haderka56442@th-nuernberg.de}
|
||||
|
||||
|
||||
\vspace{\fill}
|
||||
\indent Studiengang Wirtschaftsinformatik\\
|
||||
\indent Georg Simon Ohm\\
|
||||
\indent Technische Hochschule Nürnberg\\
|
||||
\indent Keßlerplatz 12\\
|
||||
\indent 90489 Nürnberg\\
|
||||
\indent Deutschland
|
||||
|
||||
|
||||
|
||||
\end{titlepage}
|
||||
%------------------------------------------------------------------------------
|
@ -1,62 +0,0 @@
|
||||
\relax
|
||||
\providecommand\hyper@newdestlabel[2]{}
|
||||
\providecommand\HyperFirstAtBeginDocument{\AtBeginDocument}
|
||||
\HyperFirstAtBeginDocument{\ifx\hyper@anchor\@undefined
|
||||
\global\let\oldcontentsline\contentsline
|
||||
\gdef\contentsline#1#2#3#4{\oldcontentsline{#1}{#2}{#3}}
|
||||
\global\let\oldnewlabel\newlabel
|
||||
\gdef\newlabel#1#2{\newlabelxx{#1}#2}
|
||||
\gdef\newlabelxx#1#2#3#4#5#6{\oldnewlabel{#1}{{#2}{#3}}}
|
||||
\AtEndDocument{\ifx\hyper@anchor\@undefined
|
||||
\let\contentsline\oldcontentsline
|
||||
\let\newlabel\oldnewlabel
|
||||
\fi}
|
||||
\fi}
|
||||
\global\let\hyper@last\relax
|
||||
\gdef\HyperFirstAtBeginDocument#1{#1}
|
||||
\providecommand*\HyPL@Entry[1]{}
|
||||
\catcode `"\active
|
||||
\HyPL@Entry{0<</S/D>>}
|
||||
\babel@aux{ngerman}{}
|
||||
\HyPL@Entry{1<</P()>>}
|
||||
\@writefile{toc}{\contentsline {chapter}{Declaration of Authorship}{}{dummy.1}}
|
||||
\@writefile{toc}{\contentsline {chapter}{Abstrakt}{}{dummy.2}}
|
||||
\HyPL@Entry{3<</S/D>>}
|
||||
\@writefile{toc}{\contentsline {chapter}{\numberline {1}Einleitung}{1}{chapter.3}}
|
||||
\@writefile{lof}{\addvspace {10\p@ }}
|
||||
\@writefile{lot}{\addvspace {10\p@ }}
|
||||
\@writefile{toc}{\contentsline {section}{\numberline {1.1}Ausgangsituation}{1}{section.4}}
|
||||
\@writefile{toc}{\contentsline {section}{\numberline {1.2}Ziel der Arbeit}{2}{section.5}}
|
||||
\@writefile{toc}{\contentsline {chapter}{\numberline {2}Framework}{3}{chapter.6}}
|
||||
\@writefile{lof}{\addvspace {10\p@ }}
|
||||
\@writefile{lot}{\addvspace {10\p@ }}
|
||||
\@writefile{toc}{\contentsline {section}{\numberline {2.1}Django}{3}{section.7}}
|
||||
\@writefile{toc}{\contentsline {section}{\numberline {2.2}Erweiterungen}{4}{section.8}}
|
||||
\@writefile{toc}{\contentsline {subsection}{\numberline {2.2.1}Taggable-Manager}{4}{subsection.9}}
|
||||
\@writefile{toc}{\contentsline {section}{\numberline {2.3}evtl. Bootstrap}{4}{section.10}}
|
||||
\@writefile{toc}{\contentsline {chapter}{\numberline {3}Prototyp}{5}{chapter.11}}
|
||||
\@writefile{lof}{\addvspace {10\p@ }}
|
||||
\@writefile{lot}{\addvspace {10\p@ }}
|
||||
\@writefile{toc}{\contentsline {section}{\numberline {3.1}Organisation}{5}{section.12}}
|
||||
\@writefile{toc}{\contentsline {subsection}{\numberline {3.1.1}Verwaltung im Administrator-Backend}{6}{subsection.13}}
|
||||
\@writefile{toc}{\contentsline {subsection}{\numberline {3.1.2}Berechtigungen der User}{6}{subsection.14}}
|
||||
\@writefile{toc}{\contentsline {section}{\numberline {3.2}Funktionen}{6}{section.15}}
|
||||
\@writefile{toc}{\contentsline {subsection}{\numberline {3.2.1}Abonnieren}{7}{subsection.16}}
|
||||
\@writefile{toc}{\contentsline {subsection}{\numberline {3.2.2}Filtern}{7}{subsection.17}}
|
||||
\@writefile{toc}{\contentsline {subsection}{\numberline {3.2.3}Benachrichtigungen}{7}{subsection.18}}
|
||||
\@writefile{toc}{\contentsline {chapter}{\numberline {4}Ergebnis}{8}{chapter.19}}
|
||||
\@writefile{lof}{\addvspace {10\p@ }}
|
||||
\@writefile{lot}{\addvspace {10\p@ }}
|
||||
\@writefile{toc}{\contentsline {section}{\numberline {4.1}Evaluierung}{8}{section.20}}
|
||||
\@writefile{toc}{\contentsline {chapter}{\numberline {5}Zusammenfassung und Ausblick}{9}{chapter.21}}
|
||||
\@writefile{lof}{\addvspace {10\p@ }}
|
||||
\@writefile{lot}{\addvspace {10\p@ }}
|
||||
\@writefile{toc}{\vspace {2em}}
|
||||
\@writefile{toc}{\contentsline {chapter}{\numberline {A}Anhang}{10}{appendix.22}}
|
||||
\@writefile{lof}{\addvspace {10\p@ }}
|
||||
\@writefile{lot}{\addvspace {10\p@ }}
|
||||
\@writefile{toc}{\vspace {2em}}
|
||||
\bibstyle{unsrtnat}
|
||||
\bibdata{Bibliography}
|
||||
\@writefile{toc}{\contentsline {chapter}{Literaturverzeichnis}{12}{dummy.23}}
|
||||
\newlabel{Literaturverzeichnis}{{3}{12}{Anhang}{dummy.23}{}}
|
@ -1,8 +0,0 @@
|
||||
\begin{thebibliography}{0}
|
||||
\providecommand{\natexlab}[1]{#1}
|
||||
\providecommand{\url}[1]{\texttt{#1}}
|
||||
\expandafter\ifx\csname urlstyle\endcsname\relax
|
||||
\providecommand{\doi}[1]{doi: #1}\else
|
||||
\providecommand{\doi}{doi: \begingroup \urlstyle{rm}\Url}\fi
|
||||
|
||||
\end{thebibliography}
|
@ -1,48 +0,0 @@
|
||||
This is BibTeX, Version 0.99d (TeX Live 2018)
|
||||
Capacity: max_strings=100000, hash_size=100000, hash_prime=85009
|
||||
The top-level auxiliary file: /Users/Esthi/thesis_ek/doc/graduate-thesis/.texpadtmp/Thesis.aux
|
||||
The style file: unsrtnat.bst
|
||||
I found no \citation commands---while reading file /Users/Esthi/thesis_ek/doc/graduate-thesis/.texpadtmp/Thesis.aux
|
||||
Database file #1: ../Bibliography.bib
|
||||
You've used 0 entries,
|
||||
2481 wiz_defined-function locations,
|
||||
551 strings with 4539 characters,
|
||||
and the built_in function-call counts, 34 in all, are:
|
||||
= -- 0
|
||||
> -- 0
|
||||
< -- 0
|
||||
+ -- 0
|
||||
- -- 0
|
||||
* -- 2
|
||||
:= -- 11
|
||||
add.period$ -- 0
|
||||
call.type$ -- 0
|
||||
change.case$ -- 0
|
||||
chr.to.int$ -- 0
|
||||
cite$ -- 0
|
||||
duplicate$ -- 0
|
||||
empty$ -- 1
|
||||
format.name$ -- 0
|
||||
if$ -- 1
|
||||
int.to.chr$ -- 1
|
||||
int.to.str$ -- 1
|
||||
missing$ -- 0
|
||||
newline$ -- 8
|
||||
num.names$ -- 0
|
||||
pop$ -- 0
|
||||
preamble$ -- 1
|
||||
purify$ -- 0
|
||||
quote$ -- 0
|
||||
skip$ -- 1
|
||||
stack$ -- 0
|
||||
substring$ -- 0
|
||||
swap$ -- 0
|
||||
text.length$ -- 0
|
||||
text.prefix$ -- 0
|
||||
top$ -- 0
|
||||
type$ -- 0
|
||||
warning$ -- 0
|
||||
while$ -- 0
|
||||
width$ -- 0
|
||||
write$ -- 7
|
||||
(There was 1 error message)
|
@ -1,568 +0,0 @@
|
||||
This is XeTeX, Version 3.14159265-2.6-0.99999 (TeX Live 2018) (preloaded format=xelatex 2018.6.7) 27 SEP 2018 17:21
|
||||
entering extended mode
|
||||
\write18 enabled.
|
||||
file:line:error style messages enabled.
|
||||
%&-line parsing enabled.
|
||||
**Thesis.tex
|
||||
(./Thesis.tex
|
||||
LaTeX2e <2018-04-01> patch level 2
|
||||
Babel <3.18> and hyphenation patterns for 84 language(s) loaded.
|
||||
(./Thesis.cls
|
||||
Document Class: Thesis 2007/22/02 v1.0 LaTeX document class
|
||||
(/usr/local/texlive/2018/texmf-dist/tex/latex/base/book.cls
|
||||
Document Class: book 2014/09/29 v1.4h Standard LaTeX document class
|
||||
(/usr/local/texlive/2018/texmf-dist/tex/latex/base/bk12.clo
|
||||
File: bk12.clo 2014/09/29 v1.4h Standard LaTeX file (size option)
|
||||
)
|
||||
\c@part=\count80
|
||||
\c@chapter=\count81
|
||||
\c@section=\count82
|
||||
\c@subsection=\count83
|
||||
\c@subsubsection=\count84
|
||||
\c@paragraph=\count85
|
||||
\c@subparagraph=\count86
|
||||
\c@figure=\count87
|
||||
\c@table=\count88
|
||||
\abovecaptionskip=\skip41
|
||||
\belowcaptionskip=\skip42
|
||||
\bibindent=\dimen102
|
||||
) (/usr/local/texlive/2018/texmf-dist/tex/latex/setspace/setspace.sty
|
||||
Package: setspace 2011/12/19 v6.7a set line spacing
|
||||
) (/usr/local/texlive/2018/texmf-dist/tex/latex/vmargin/vmargin.sty
|
||||
Package: vmargin 2004/07/15 V2.5 set document margins (VK)
|
||||
|
||||
Package: vmargin 2004/07/15 V2.5 set document margins (VK)
|
||||
\PaperWidth=\dimen103
|
||||
\PaperHeight=\dimen104
|
||||
) (/usr/local/texlive/2018/texmf-dist/tex/latex/fancyhdr/fancyhdr.sty
|
||||
Package: fancyhdr 2017/06/30 v3.9a Extensive control of page headers and footers
|
||||
\f@nch@headwidth=\skip43
|
||||
\f@nch@O@elh=\skip44
|
||||
\f@nch@O@erh=\skip45
|
||||
\f@nch@O@olh=\skip46
|
||||
\f@nch@O@orh=\skip47
|
||||
\f@nch@O@elf=\skip48
|
||||
\f@nch@O@erf=\skip49
|
||||
\f@nch@O@olf=\skip50
|
||||
\f@nch@O@orf=\skip51
|
||||
) (/usr/local/texlive/2018/texmf-dist/tex/latex/amsmath/amsmath.sty
|
||||
Package: amsmath 2017/09/02 v2.17a AMS math features
|
||||
\@mathmargin=\skip52
|
||||
|
||||
For additional information on amsmath, use the `?' option.
|
||||
(/usr/local/texlive/2018/texmf-dist/tex/latex/amsmath/amstext.sty
|
||||
Package: amstext 2000/06/29 v2.01 AMS text
|
||||
(/usr/local/texlive/2018/texmf-dist/tex/latex/amsmath/amsgen.sty
|
||||
File: amsgen.sty 1999/11/30 v2.0 generic functions
|
||||
\@emptytoks=\toks14
|
||||
\ex@=\dimen105
|
||||
)) (/usr/local/texlive/2018/texmf-dist/tex/latex/amsmath/amsbsy.sty
|
||||
Package: amsbsy 1999/11/29 v1.2d Bold Symbols
|
||||
\pmbraise@=\dimen106
|
||||
) (/usr/local/texlive/2018/texmf-dist/tex/latex/amsmath/amsopn.sty
|
||||
Package: amsopn 2016/03/08 v2.02 operator names
|
||||
)
|
||||
\inf@bad=\count89
|
||||
LaTeX Info: Redefining \frac on input line 213.
|
||||
\uproot@=\count90
|
||||
\leftroot@=\count91
|
||||
LaTeX Info: Redefining \overline on input line 375.
|
||||
\classnum@=\count92
|
||||
\DOTSCASE@=\count93
|
||||
LaTeX Info: Redefining \ldots on input line 472.
|
||||
LaTeX Info: Redefining \dots on input line 475.
|
||||
LaTeX Info: Redefining \cdots on input line 596.
|
||||
\Mathstrutbox@=\box26
|
||||
\strutbox@=\box27
|
||||
\big@size=\dimen107
|
||||
LaTeX Font Info: Redeclaring font encoding OML on input line 712.
|
||||
LaTeX Font Info: Redeclaring font encoding OMS on input line 713.
|
||||
\macc@depth=\count94
|
||||
\c@MaxMatrixCols=\count95
|
||||
\dotsspace@=\muskip10
|
||||
\c@parentequation=\count96
|
||||
\dspbrk@lvl=\count97
|
||||
\tag@help=\toks15
|
||||
\row@=\count98
|
||||
\column@=\count99
|
||||
\maxfields@=\count100
|
||||
\andhelp@=\toks16
|
||||
\eqnshift@=\dimen108
|
||||
\alignsep@=\dimen109
|
||||
\tagshift@=\dimen110
|
||||
\tagwidth@=\dimen111
|
||||
\totwidth@=\dimen112
|
||||
\lineht@=\dimen113
|
||||
\@envbody=\toks17
|
||||
\multlinegap=\skip53
|
||||
\multlinetaggap=\skip54
|
||||
\mathdisplay@stack=\toks18
|
||||
LaTeX Info: Redefining \[ on input line 2817.
|
||||
LaTeX Info: Redefining \] on input line 2818.
|
||||
) (/usr/local/texlive/2018/texmf-dist/tex/latex/amsfonts/amsfonts.sty
|
||||
Package: amsfonts 2013/01/14 v3.01 Basic AMSFonts support
|
||||
\symAMSa=\mathgroup4
|
||||
\symAMSb=\mathgroup5
|
||||
LaTeX Font Info: Overwriting math alphabet `\mathfrak' in version `bold'
|
||||
(Font) U/euf/m/n --> U/euf/b/n on input line 106.
|
||||
) (/usr/local/texlive/2018/texmf-dist/tex/latex/amsfonts/amssymb.sty
|
||||
Package: amssymb 2013/01/14 v3.01 AMS font symbols
|
||||
) (/usr/local/texlive/2018/texmf-dist/tex/latex/amsmath/amscd.sty
|
||||
Package: amscd 2017/04/14 v2.1 AMS Commutative Diagrams
|
||||
\athelp@=\toks19
|
||||
\minaw@=\dimen114
|
||||
\bigaw@=\dimen115
|
||||
\minCDarrowwidth=\dimen116
|
||||
) (/usr/local/texlive/2018/texmf-dist/tex/latex/amscls/amsthm.sty
|
||||
Package: amsthm 2017/10/31 v2.20.4
|
||||
\thm@style=\toks20
|
||||
\thm@bodyfont=\toks21
|
||||
\thm@headfont=\toks22
|
||||
\thm@notefont=\toks23
|
||||
\thm@headpunct=\toks24
|
||||
\thm@preskip=\skip55
|
||||
\thm@postskip=\skip56
|
||||
\thm@headsep=\skip57
|
||||
\dth@everypar=\toks25
|
||||
) (/usr/local/texlive/2018/texmf-dist/tex/latex/tools/xspace.sty
|
||||
Package: xspace 2014/10/28 v1.13 Space after command names (DPC,MH)
|
||||
)
|
||||
\c@example=\count101
|
||||
\c@theorem=\count102
|
||||
(/usr/local/texlive/2018/texmf-dist/tex/latex/caption/caption.sty
|
||||
Package: caption 2016/02/21 v3.3-144 Customizing captions (AR)
|
||||
(/usr/local/texlive/2018/texmf-dist/tex/latex/caption/caption3.sty
|
||||
Package: caption3 2016/05/22 v1.7-166 caption3 kernel (AR)
|
||||
Package caption3 Info: TeX engine: e-TeX on input line 67.
|
||||
(/usr/local/texlive/2018/texmf-dist/tex/latex/graphics/keyval.sty
|
||||
Package: keyval 2014/10/28 v1.15 key=value parser (DPC)
|
||||
\KV@toks@=\toks26
|
||||
)
|
||||
\captionmargin=\dimen117
|
||||
\captionmargin@=\dimen118
|
||||
\captionwidth=\dimen119
|
||||
\caption@tempdima=\dimen120
|
||||
\caption@indent=\dimen121
|
||||
\caption@parindent=\dimen122
|
||||
\caption@hangindent=\dimen123
|
||||
)
|
||||
\c@ContinuedFloat=\count103
|
||||
) (/usr/local/texlive/2018/texmf-dist/tex/latex/graphics/graphicx.sty
|
||||
Package: graphicx 2017/06/01 v1.1a Enhanced LaTeX Graphics (DPC,SPQR)
|
||||
(/usr/local/texlive/2018/texmf-dist/tex/latex/graphics/graphics.sty
|
||||
Package: graphics 2017/06/25 v1.2c Standard LaTeX Graphics (DPC,SPQR)
|
||||
(/usr/local/texlive/2018/texmf-dist/tex/latex/graphics/trig.sty
|
||||
Package: trig 2016/01/03 v1.10 sin cos tan (DPC)
|
||||
) (/usr/local/texlive/2018/texmf-dist/tex/latex/graphics-cfg/graphics.cfg
|
||||
File: graphics.cfg 2016/06/04 v1.11 sample graphics configuration
|
||||
)
|
||||
Package graphics Info: Driver file: xetex.def on input line 99.
|
||||
(/usr/local/texlive/2018/texmf-dist/tex/latex/graphics-def/xetex.def
|
||||
File: xetex.def 2017/06/24 v5.0h Graphics/color driver for xetex
|
||||
))
|
||||
\Gin@req@height=\dimen124
|
||||
\Gin@req@width=\dimen125
|
||||
) (/usr/local/texlive/2018/texmf-dist/tex/latex/oberdiek/epstopdf.sty
|
||||
Package: epstopdf 2016/05/15 v2.6 Conversion with epstopdf on the fly (HO)
|
||||
(/usr/local/texlive/2018/texmf-dist/tex/latex/oberdiek/epstopdf-base.sty
|
||||
Package: epstopdf-base 2016/05/15 v2.6 Base part for package epstopdf
|
||||
(/usr/local/texlive/2018/texmf-dist/tex/generic/oberdiek/infwarerr.sty
|
||||
Package: infwarerr 2016/05/16 v1.4 Providing info/warning/error messages (HO)
|
||||
) (/usr/local/texlive/2018/texmf-dist/tex/latex/oberdiek/grfext.sty
|
||||
Package: grfext 2016/05/16 v1.2 Manage graphics extensions (HO)
|
||||
(/usr/local/texlive/2018/texmf-dist/tex/generic/oberdiek/kvdefinekeys.sty
|
||||
Package: kvdefinekeys 2016/05/16 v1.4 Define keys (HO)
|
||||
(/usr/local/texlive/2018/texmf-dist/tex/generic/oberdiek/ltxcmds.sty
|
||||
Package: ltxcmds 2016/05/16 v1.23 LaTeX kernel commands for general use (HO)
|
||||
))) (/usr/local/texlive/2018/texmf-dist/tex/latex/oberdiek/kvoptions.sty
|
||||
Package: kvoptions 2016/05/16 v3.12 Key value format for package options (HO)
|
||||
(/usr/local/texlive/2018/texmf-dist/tex/generic/oberdiek/kvsetkeys.sty
|
||||
Package: kvsetkeys 2016/05/16 v1.17 Key value parser (HO)
|
||||
(/usr/local/texlive/2018/texmf-dist/tex/generic/oberdiek/etexcmds.sty
|
||||
Package: etexcmds 2016/05/16 v1.6 Avoid name clashes with e-TeX commands (HO)
|
||||
(/usr/local/texlive/2018/texmf-dist/tex/generic/oberdiek/ifluatex.sty
|
||||
Package: ifluatex 2016/05/16 v1.4 Provides the ifluatex switch (HO)
|
||||
Package ifluatex Info: LuaTeX not detected.
|
||||
)
|
||||
Package etexcmds Info: Could not find \expanded.
|
||||
(etexcmds) That can mean that you are not using pdfTeX 1.50 or
|
||||
(etexcmds) that some package has redefined \expanded.
|
||||
(etexcmds) In the latter case, load this package earlier.
|
||||
))) (/usr/local/texlive/2018/texmf-dist/tex/generic/oberdiek/pdftexcmds.sty
|
||||
Package: pdftexcmds 2018/01/30 v0.27 Utility functions of pdfTeX for LuaTeX (HO)
|
||||
(/usr/local/texlive/2018/texmf-dist/tex/generic/oberdiek/ifpdf.sty
|
||||
Package: ifpdf 2017/03/15 v3.2 Provides the ifpdf switch
|
||||
)
|
||||
Package pdftexcmds Info: LuaTeX not detected.
|
||||
Package pdftexcmds Info: pdfTeX >= 1.30 not detected.
|
||||
Package pdftexcmds Info: \pdf@primitive is available.
|
||||
Package pdftexcmds Info: \pdf@ifprimitive is available.
|
||||
Package pdftexcmds Info: \pdfdraftmode not found.
|
||||
)
|
||||
|
||||
Package epstopdf Warning: Drivers other than `pdftex' and `luatex' are not supported.
|
||||
|
||||
)) (/usr/local/texlive/2018/texmf-dist/tex/latex/subfigure/subfigure.sty
|
||||
Package: subfigure 2002/03/15 v2.1.5 subfigure package
|
||||
\subfigtopskip=\skip58
|
||||
\subfigcapskip=\skip59
|
||||
\subfigcaptopadj=\dimen126
|
||||
\subfigbottomskip=\skip60
|
||||
\subfigcapmargin=\dimen127
|
||||
\subfiglabelskip=\skip61
|
||||
\c@subfigure=\count104
|
||||
\c@lofdepth=\count105
|
||||
\c@subtable=\count106
|
||||
\c@lotdepth=\count107
|
||||
|
||||
****************************************
|
||||
* Local config file subfigure.cfg used *
|
||||
****************************************
|
||||
(/usr/local/texlive/2018/texmf-dist/tex/latex/subfigure/subfigure.cfg)
|
||||
\subfig@top=\skip62
|
||||
\subfig@bottom=\skip63
|
||||
) (/usr/local/texlive/2018/texmf-dist/tex/latex/booktabs/booktabs.sty
|
||||
Package: booktabs 2016/04/27 v1.618033 publication quality tables
|
||||
\heavyrulewidth=\dimen128
|
||||
\lightrulewidth=\dimen129
|
||||
\cmidrulewidth=\dimen130
|
||||
\belowrulesep=\dimen131
|
||||
\belowbottomsep=\dimen132
|
||||
\aboverulesep=\dimen133
|
||||
\abovetopsep=\dimen134
|
||||
\cmidrulesep=\dimen135
|
||||
\cmidrulekern=\dimen136
|
||||
\defaultaddspace=\dimen137
|
||||
\@cmidla=\count108
|
||||
\@cmidlb=\count109
|
||||
\@aboverulesep=\dimen138
|
||||
\@belowrulesep=\dimen139
|
||||
\@thisruleclass=\count110
|
||||
\@lastruleclass=\count111
|
||||
\@thisrulewidth=\dimen140
|
||||
) (/usr/local/texlive/2018/texmf-dist/tex/latex/graphics/rotating.sty
|
||||
Package: rotating 2016/08/11 v2.16d rotated objects in LaTeX
|
||||
(/usr/local/texlive/2018/texmf-dist/tex/latex/base/ifthen.sty
|
||||
Package: ifthen 2014/09/29 v1.1c Standard LaTeX ifthen package (DPC)
|
||||
)
|
||||
\c@r@tfl@t=\count112
|
||||
\rotFPtop=\skip64
|
||||
\rotFPbot=\skip65
|
||||
\rot@float@box=\box28
|
||||
\rot@mess@toks=\toks27
|
||||
) (/usr/local/texlive/2018/texmf-dist/tex/latex/listings/listings.sty
|
||||
\lst@mode=\count113
|
||||
\lst@gtempboxa=\box29
|
||||
\lst@token=\toks28
|
||||
\lst@length=\count114
|
||||
\lst@currlwidth=\dimen141
|
||||
\lst@column=\count115
|
||||
\lst@pos=\count116
|
||||
\lst@lostspace=\dimen142
|
||||
\lst@width=\dimen143
|
||||
\lst@newlines=\count117
|
||||
\lst@lineno=\count118
|
||||
\lst@maxwidth=\dimen144
|
||||
(./lstpatch.sty) (/usr/local/texlive/2018/texmf-dist/tex/latex/listings/lstmisc.sty
|
||||
File: lstmisc.sty 2015/06/04 1.6 (Carsten Heinz)
|
||||
\c@lstnumber=\count119
|
||||
\lst@skipnumbers=\count120
|
||||
\lst@framebox=\box30
|
||||
) (/usr/local/texlive/2018/texmf-dist/tex/latex/listings/listings.cfg
|
||||
File: listings.cfg 2015/06/04 1.6 listings configuration
|
||||
))
|
||||
Package: listings 2015/06/04 1.6 (Carsten Heinz)
|
||||
(./lstpatch.sty) (/usr/local/texlive/2018/texmf-dist/tex/latex/hyperref/hyperref.sty
|
||||
Package: hyperref 2018/02/06 v6.86b Hypertext links for LaTeX
|
||||
(/usr/local/texlive/2018/texmf-dist/tex/generic/oberdiek/hobsub-hyperref.sty
|
||||
Package: hobsub-hyperref 2016/05/16 v1.14 Bundle oberdiek, subset hyperref (HO)
|
||||
(/usr/local/texlive/2018/texmf-dist/tex/generic/oberdiek/hobsub-generic.sty
|
||||
Package: hobsub-generic 2016/05/16 v1.14 Bundle oberdiek, subset generic (HO)
|
||||
Package: hobsub 2016/05/16 v1.14 Construct package bundles (HO)
|
||||
Package hobsub Info: Skipping package `infwarerr' (already loaded).
|
||||
Package hobsub Info: Skipping package `ltxcmds' (already loaded).
|
||||
Package hobsub Info: Skipping package `ifluatex' (already loaded).
|
||||
Package: ifvtex 2016/05/16 v1.6 Detect VTeX and its facilities (HO)
|
||||
Package ifvtex Info: VTeX not detected.
|
||||
Package: intcalc 2016/05/16 v1.2 Expandable calculations with integers (HO)
|
||||
Package hobsub Info: Skipping package `ifpdf' (already loaded).
|
||||
Package hobsub Info: Skipping package `etexcmds' (already loaded).
|
||||
Package hobsub Info: Skipping package `kvsetkeys' (already loaded).
|
||||
Package hobsub Info: Skipping package `kvdefinekeys' (already loaded).
|
||||
Package hobsub Info: Skipping package `pdftexcmds' (already loaded).
|
||||
Package: pdfescape 2016/05/16 v1.14 Implements pdfTeX's escape features (HO)
|
||||
Package: bigintcalc 2016/05/16 v1.4 Expandable calculations on big integers (HO)
|
||||
Package: bitset 2016/05/16 v1.2 Handle bit-vector datatype (HO)
|
||||
Package: uniquecounter 2016/05/16 v1.3 Provide unlimited unique counter (HO)
|
||||
)
|
||||
Package hobsub Info: Skipping package `hobsub' (already loaded).
|
||||
Package: letltxmacro 2016/05/16 v1.5 Let assignment for LaTeX macros (HO)
|
||||
Package: hopatch 2016/05/16 v1.3 Wrapper for package hooks (HO)
|
||||
Package: xcolor-patch 2016/05/16 xcolor patch
|
||||
Package: atveryend 2016/05/16 v1.9 Hooks at the very end of document (HO)
|
||||
Package atveryend Info: \enddocument detected (standard20110627).
|
||||
Package: atbegshi 2016/06/09 v1.18 At begin shipout hook (HO)
|
||||
Package: refcount 2016/05/16 v3.5 Data extraction from label references (HO)
|
||||
Package: hycolor 2016/05/16 v1.8 Color options for hyperref/bookmark (HO)
|
||||
) (/usr/local/texlive/2018/texmf-dist/tex/generic/ifxetex/ifxetex.sty
|
||||
Package: ifxetex 2010/09/12 v0.6 Provides ifxetex conditional
|
||||
) (/usr/local/texlive/2018/texmf-dist/tex/latex/oberdiek/auxhook.sty
|
||||
Package: auxhook 2016/05/16 v1.4 Hooks for auxiliary files (HO)
|
||||
)
|
||||
\@linkdim=\dimen145
|
||||
\Hy@linkcounter=\count121
|
||||
\Hy@pagecounter=\count122
|
||||
(/usr/local/texlive/2018/texmf-dist/tex/latex/hyperref/pd1enc.def
|
||||
File: pd1enc.def 2018/02/06 v6.86b Hyperref: PDFDocEncoding definition (HO)
|
||||
)
|
||||
\Hy@SavedSpaceFactor=\count123
|
||||
(/usr/local/texlive/2018/texmf-dist/tex/latex/latexconfig/hyperref.cfg
|
||||
File: hyperref.cfg 2002/06/06 v1.2 hyperref configuration of TeXLive
|
||||
)
|
||||
Package hyperref Info: Option `bookmarks' set `true' on input line 4383.
|
||||
Package hyperref Info: Option `bookmarksopen' set `true' on input line 4383.
|
||||
Package hyperref Info: Option `bookmarksnumbered' set `true' on input line 4383.
|
||||
Package hyperref Info: Option `hypertexnames' set `false' on input line 4383.
|
||||
Package hyperref Info: Option `colorlinks' set `true' on input line 4383.
|
||||
Package hyperref Info: Option `unicode' set `true' on input line 4383.
|
||||
(/usr/local/texlive/2018/texmf-dist/tex/latex/hyperref/puenc.def
|
||||
File: puenc.def 2018/02/06 v6.86b Hyperref: PDF Unicode definition (HO)
|
||||
)
|
||||
Package hyperref Info: Option `breaklinks' set `true' on input line 4383.
|
||||
Package hyperref Info: Hyper figures OFF on input line 4509.
|
||||
Package hyperref Info: Link nesting OFF on input line 4514.
|
||||
Package hyperref Info: Hyper index ON on input line 4517.
|
||||
Package hyperref Info: Plain pages OFF on input line 4524.
|
||||
Package hyperref Info: Backreferencing OFF on input line 4529.
|
||||
Package hyperref Info: Implicit mode ON; LaTeX internals redefined.
|
||||
Package hyperref Info: Bookmarks ON on input line 4762.
|
||||
\c@Hy@tempcnt=\count124
|
||||
(/usr/local/texlive/2018/texmf-dist/tex/latex/url/url.sty
|
||||
\Urlmuskip=\muskip11
|
||||
Package: url 2013/09/16 ver 3.4 Verb mode for urls, etc.
|
||||
)
|
||||
LaTeX Info: Redefining \url on input line 5115.
|
||||
\XeTeXLinkMargin=\dimen146
|
||||
\Fld@menulength=\count125
|
||||
\Field@Width=\dimen147
|
||||
\Fld@charsize=\dimen148
|
||||
Package hyperref Info: Hyper figures OFF on input line 6369.
|
||||
Package hyperref Info: Link nesting OFF on input line 6374.
|
||||
Package hyperref Info: Hyper index ON on input line 6377.
|
||||
Package hyperref Info: backreferencing OFF on input line 6384.
|
||||
Package hyperref Info: Link coloring ON on input line 6387.
|
||||
Package hyperref Info: Link coloring with OCG OFF on input line 6394.
|
||||
Package hyperref Info: PDF/A mode OFF on input line 6399.
|
||||
LaTeX Info: Redefining \ref on input line 6439.
|
||||
LaTeX Info: Redefining \pageref on input line 6443.
|
||||
\Hy@abspage=\count126
|
||||
\c@Item=\count127
|
||||
\c@Hfootnote=\count128
|
||||
)
|
||||
Package hyperref Info: Driver (autodetected): hxetex.
|
||||
(/usr/local/texlive/2018/texmf-dist/tex/latex/hyperref/hxetex.def
|
||||
File: hxetex.def 2018/02/06 v6.86b Hyperref driver for XeTeX
|
||||
(/usr/local/texlive/2018/texmf-dist/tex/generic/oberdiek/stringenc.sty
|
||||
Package: stringenc 2016/05/16 v1.11 Convert strings between diff. encodings (HO)
|
||||
)
|
||||
\pdfm@box=\box31
|
||||
\c@Hy@AnnotLevel=\count129
|
||||
\HyField@AnnotCount=\count130
|
||||
\Fld@listcount=\count131
|
||||
\c@bookmark@seq@number=\count132
|
||||
(/usr/local/texlive/2018/texmf-dist/tex/latex/oberdiek/rerunfilecheck.sty
|
||||
Package: rerunfilecheck 2016/05/16 v1.8 Rerun checks for auxiliary files (HO)
|
||||
Package uniquecounter Info: New unique counter `rerunfilecheck' on input line 282.
|
||||
)
|
||||
\Hy@SectionHShift=\skip66
|
||||
)
|
||||
\c@dummy=\count133
|
||||
) (/usr/local/texlive/2018/texmf-dist/tex/latex/titlesec/titlesec.sty
|
||||
Package: titlesec 2016/03/21 v2.10.2 Sectioning titles
|
||||
\ttl@box=\box32
|
||||
\beforetitleunit=\skip67
|
||||
\aftertitleunit=\skip68
|
||||
\ttl@plus=\dimen149
|
||||
\ttl@minus=\dimen150
|
||||
\ttl@toksa=\toks29
|
||||
\titlewidth=\dimen151
|
||||
\titlewidthlast=\dimen152
|
||||
\titlewidthfirst=\dimen153
|
||||
) (/usr/local/texlive/2018/texmf-dist/tex/generic/babel/babel.sty
|
||||
Package: babel 2018/02/14 3.18 The Babel package
|
||||
(/usr/local/texlive/2018/texmf-dist/tex/generic/babel/switch.def
|
||||
File: switch.def 2018/02/14 3.18 Babel switching mechanism
|
||||
) (/usr/local/texlive/2018/texmf-dist/tex/generic/babel-german/ngerman.ldf
|
||||
Language: ngerman 2018/03/29 v2.10 German support for babel (post-1996 orthography)
|
||||
(/usr/local/texlive/2018/texmf-dist/tex/generic/babel-german/ngermanb.ldf
|
||||
Language: ngermanb 2018/03/29 v2.10 German support for babel (post-1996 orthography)
|
||||
(/usr/local/texlive/2018/texmf-dist/tex/generic/babel/babel.def
|
||||
File: babel.def 2018/02/14 3.18 Babel common definitions
|
||||
\babel@savecnt=\count134
|
||||
\U@D=\dimen154
|
||||
(/usr/local/texlive/2018/texmf-dist/tex/generic/babel/xebabel.def (/usr/local/texlive/2018/texmf-dist/tex/generic/babel/txtbabel.def))
|
||||
\bbl@dirlevel=\count135
|
||||
)
|
||||
Package babel Info: Making " an active character on input line 121.
|
||||
)))
|
||||
Package hyperref Info: Option `colorlinks' set `true' on input line 15.
|
||||
(/Users/Esthi/thesis_ek/doc/graduate-thesis/.texpadtmp/Thesis.aux)
|
||||
\openout1 = `Thesis.aux'.
|
||||
|
||||
LaTeX Font Info: Checking defaults for OML/cmm/m/it on input line 19.
|
||||
LaTeX Font Info: ... okay on input line 19.
|
||||
LaTeX Font Info: Checking defaults for T1/cmr/m/n on input line 19.
|
||||
LaTeX Font Info: ... okay on input line 19.
|
||||
LaTeX Font Info: Checking defaults for OT1/cmr/m/n on input line 19.
|
||||
LaTeX Font Info: ... okay on input line 19.
|
||||
LaTeX Font Info: Checking defaults for OMS/cmsy/m/n on input line 19.
|
||||
LaTeX Font Info: ... okay on input line 19.
|
||||
LaTeX Font Info: Checking defaults for TU/lmr/m/n on input line 19.
|
||||
LaTeX Font Info: ... okay on input line 19.
|
||||
LaTeX Font Info: Checking defaults for OMX/cmex/m/n on input line 19.
|
||||
LaTeX Font Info: ... okay on input line 19.
|
||||
LaTeX Font Info: Checking defaults for U/cmr/m/n on input line 19.
|
||||
LaTeX Font Info: ... okay on input line 19.
|
||||
LaTeX Font Info: Checking defaults for PD1/pdf/m/n on input line 19.
|
||||
LaTeX Font Info: ... okay on input line 19.
|
||||
LaTeX Font Info: Checking defaults for PU/pdf/m/n on input line 19.
|
||||
LaTeX Font Info: ... okay on input line 19.
|
||||
Package caption Info: Begin \AtBeginDocument code.
|
||||
Package caption Info: hyperref package is loaded.
|
||||
Package caption Info: listings package is loaded.
|
||||
Package caption Info: rotating package is loaded.
|
||||
Package caption Info: subfigure package is loaded.
|
||||
Package caption Info: End \AtBeginDocument code.
|
||||
\c@lstlisting=\count136
|
||||
\AtBeginShipoutBox=\box33
|
||||
(/usr/local/texlive/2018/texmf-dist/tex/latex/graphics/color.sty
|
||||
Package: color 2016/07/10 v1.1e Standard LaTeX Color (DPC)
|
||||
|
||||
(/usr/local/texlive/2018/texmf-dist/tex/latex/graphics-cfg/color.cfg
|
||||
File: color.cfg 2016/01/02 v1.6 sample color configuration
|
||||
)
|
||||
Package color Info: Driver file: xetex.def on input line 147.
|
||||
)
|
||||
Package hyperref Info: Link coloring ON on input line 19.
|
||||
(/usr/local/texlive/2018/texmf-dist/tex/latex/hyperref/nameref.sty
|
||||
Package: nameref 2016/05/21 v2.44 Cross-referencing by name of section
|
||||
(/usr/local/texlive/2018/texmf-dist/tex/generic/oberdiek/gettitlestring.sty
|
||||
Package: gettitlestring 2016/05/16 v1.5 Cleanup title references (HO)
|
||||
)
|
||||
\c@section@level=\count137
|
||||
)
|
||||
LaTeX Info: Redefining \ref on input line 19.
|
||||
LaTeX Info: Redefining \pageref on input line 19.
|
||||
LaTeX Info: Redefining \nameref on input line 19.
|
||||
(/Users/Esthi/thesis_ek/doc/graduate-thesis/.texpadtmp/Thesis.out) (/Users/Esthi/thesis_ek/doc/graduate-thesis/.texpadtmp/Thesis.out)
|
||||
\@outlinefile=\write3
|
||||
\openout3 = `Thesis.out'.
|
||||
|
||||
|
||||
------------------------------------------------------------------------------
|
||||
Title Page
|
||||
------------------------------------------------------------------------------
|
||||
[1
|
||||
|
||||
|
||||
]
|
||||
------------------------------------------------------------------------------
|
||||
Declaration of Authorship
|
||||
------------------------------------------------------------------------------
|
||||
LaTeX Font Info: Try loading font information for U+msa on input line 74.
|
||||
(/usr/local/texlive/2018/texmf-dist/tex/latex/amsfonts/umsa.fd
|
||||
File: umsa.fd 2013/01/14 v3.01 AMS symbols A
|
||||
)
|
||||
LaTeX Font Info: Try loading font information for U+msb on input line 74.
|
||||
(/usr/local/texlive/2018/texmf-dist/tex/latex/amsfonts/umsb.fd
|
||||
File: umsb.fd 2013/01/14 v3.01 AMS symbols B
|
||||
)
|
||||
Underfull \hbox (badness 10000) in paragraph at lines 74--74
|
||||
|
||||
[]
|
||||
|
||||
[1
|
||||
|
||||
]
|
||||
------------------------------------------------------------------------------
|
||||
Abstract Page
|
||||
------------------------------------------------------------------------------
|
||||
[2
|
||||
|
||||
] (./Chapters/Chapter1.tex
|
||||
Kapitel 1.
|
||||
------------------------------------------------------------------------------
|
||||
1 Einleitung
|
||||
------------------------------------------------------------------------------
|
||||
) [1
|
||||
|
||||
|
||||
|
||||
] (./Chapters/Chapter2.tex [2]
|
||||
Kapitel 2.
|
||||
------------------------------------------------------------------------------
|
||||
2 Framework
|
||||
------------------------------------------------------------------------------
|
||||
[3
|
||||
|
||||
]) (./Chapters/Chapter3.tex [4]
|
||||
Kapitel 3.
|
||||
------------------------------------------------------------------------------
|
||||
3 Prototyp
|
||||
------------------------------------------------------------------------------
|
||||
[5
|
||||
|
||||
] [6]) (./Chapters/Chapter4.tex [7]
|
||||
Kapitel 4.
|
||||
------------------------------------------------------------------------------
|
||||
4 Ergebnis
|
||||
------------------------------------------------------------------------------
|
||||
) (./Chapters/Chapter5.tex [8
|
||||
|
||||
]
|
||||
Kapitel 5.
|
||||
------------------------------------------------------------------------------
|
||||
5 Zusammenfassung und Ausblick
|
||||
------------------------------------------------------------------------------
|
||||
) (./Appendices/AppendixA.tex [9
|
||||
|
||||
]
|
||||
Anhang A.
|
||||
------------------------------------------------------------------------------
|
||||
A Anhang
|
||||
------------------------------------------------------------------------------
|
||||
[10
|
||||
|
||||
]) [11]
|
||||
------------------------------------------------------------------------------
|
||||
Literaturverzeichnis
|
||||
------------------------------------------------------------------------------
|
||||
(/Users/Esthi/thesis_ek/doc/graduate-thesis/.texpadtmp/Thesis.bbl
|
||||
|
||||
LaTeX Warning: Empty `thebibliography' environment on input line 8.
|
||||
|
||||
)
|
||||
Package atveryend Info: Empty hook `BeforeClearDocument' on input line 139.
|
||||
[12
|
||||
|
||||
|
||||
]
|
||||
Package atveryend Info: Empty hook `AfterLastShipout' on input line 139.
|
||||
(/Users/Esthi/thesis_ek/doc/graduate-thesis/.texpadtmp/Thesis.aux)
|
||||
Package atveryend Info: Empty hook `AtVeryEndDocument' on input line 139.
|
||||
Package atveryend Info: Executing hook `AtEndAfterFileList' on input line 139.
|
||||
Package rerunfilecheck Info: File `Thesis.out' has not changed.
|
||||
(rerunfilecheck) Checksum: .
|
||||
Package atveryend Info: Empty hook `AtVeryVeryEnd' on input line 139.
|
||||
)
|
||||
(\end occurred when \ifx on line 161 was incomplete)
|
||||
Here is how much of TeX's memory you used:
|
||||
12784 strings out of 492970
|
||||
183565 string characters out of 6133939
|
||||
282101 words of memory out of 5000000
|
||||
16599 multiletter control sequences out of 15000+600000
|
||||
6282 words of font info for 34 fonts, out of 8000000 for 9000
|
||||
1348 hyphenation exceptions out of 8191
|
||||
49i,11n,45p,13111b,360s stack positions out of 5000i,500n,10000p,200000b,80000s
|
||||
|
||||
Output written on /Users/Esthi/thesis_ek/doc/graduate-thesis/.texpadtmp/Thesis.pdf (15 pages).
|
@ -1,23 +0,0 @@
|
||||
\BOOKMARK [0][-]{dummy.1}{\376\377\000D\000e\000c\000l\000a\000r\000a\000t\000i\000o\000n\000\040\000o\000f\000\040\000A\000u\000t\000h\000o\000r\000s\000h\000i\000p}{}% 1
|
||||
\BOOKMARK [0][-]{dummy.2}{\376\377\000A\000b\000s\000t\000r\000a\000k\000t}{}% 2
|
||||
\BOOKMARK [0][-]{chapter.3}{\376\377\0001\000\040\000E\000i\000n\000l\000e\000i\000t\000u\000n\000g}{}% 3
|
||||
\BOOKMARK [1][-]{section.4}{\376\377\0001\000.\0001\000\040\000A\000u\000s\000g\000a\000n\000g\000s\000i\000t\000u\000a\000t\000i\000o\000n}{chapter.3}% 4
|
||||
\BOOKMARK [1][-]{section.5}{\376\377\0001\000.\0002\000\040\000Z\000i\000e\000l\000\040\000d\000e\000r\000\040\000A\000r\000b\000e\000i\000t}{chapter.3}% 5
|
||||
\BOOKMARK [0][-]{chapter.6}{\376\377\0002\000\040\000F\000r\000a\000m\000e\000w\000o\000r\000k}{}% 6
|
||||
\BOOKMARK [1][-]{section.7}{\376\377\0002\000.\0001\000\040\000D\000j\000a\000n\000g\000o}{chapter.6}% 7
|
||||
\BOOKMARK [1][-]{section.8}{\376\377\0002\000.\0002\000\040\000E\000r\000w\000e\000i\000t\000e\000r\000u\000n\000g\000e\000n}{chapter.6}% 8
|
||||
\BOOKMARK [2][-]{subsection.9}{\376\377\0002\000.\0002\000.\0001\000\040\000T\000a\000g\000g\000a\000b\000l\000e\000-\000M\000a\000n\000a\000g\000e\000r}{section.8}% 9
|
||||
\BOOKMARK [1][-]{section.10}{\376\377\0002\000.\0003\000\040\000e\000v\000t\000l\000.\000\040\000B\000o\000o\000t\000s\000t\000r\000a\000p}{chapter.6}% 10
|
||||
\BOOKMARK [0][-]{chapter.11}{\376\377\0003\000\040\000P\000r\000o\000t\000o\000t\000y\000p}{}% 11
|
||||
\BOOKMARK [1][-]{section.12}{\376\377\0003\000.\0001\000\040\000O\000r\000g\000a\000n\000i\000s\000a\000t\000i\000o\000n}{chapter.11}% 12
|
||||
\BOOKMARK [2][-]{subsection.13}{\376\377\0003\000.\0001\000.\0001\000\040\000V\000e\000r\000w\000a\000l\000t\000u\000n\000g\000\040\000i\000m\000\040\000A\000d\000m\000i\000n\000i\000s\000t\000r\000a\000t\000o\000r\000-\000B\000a\000c\000k\000e\000n\000d}{section.12}% 13
|
||||
\BOOKMARK [2][-]{subsection.14}{\376\377\0003\000.\0001\000.\0002\000\040\000B\000e\000r\000e\000c\000h\000t\000i\000g\000u\000n\000g\000e\000n\000\040\000d\000e\000r\000\040\000U\000s\000e\000r}{section.12}% 14
|
||||
\BOOKMARK [1][-]{section.15}{\376\377\0003\000.\0002\000\040\000F\000u\000n\000k\000t\000i\000o\000n\000e\000n}{chapter.11}% 15
|
||||
\BOOKMARK [2][-]{subsection.16}{\376\377\0003\000.\0002\000.\0001\000\040\000A\000b\000o\000n\000n\000i\000e\000r\000e\000n}{section.15}% 16
|
||||
\BOOKMARK [2][-]{subsection.17}{\376\377\0003\000.\0002\000.\0002\000\040\000F\000i\000l\000t\000e\000r\000n}{section.15}% 17
|
||||
\BOOKMARK [2][-]{subsection.18}{\376\377\0003\000.\0002\000.\0003\000\040\000B\000e\000n\000a\000c\000h\000r\000i\000c\000h\000t\000i\000g\000u\000n\000g\000e\000n}{section.15}% 18
|
||||
\BOOKMARK [0][-]{chapter.19}{\376\377\0004\000\040\000E\000r\000g\000e\000b\000n\000i\000s}{}% 19
|
||||
\BOOKMARK [1][-]{section.20}{\376\377\0004\000.\0001\000\040\000E\000v\000a\000l\000u\000i\000e\000r\000u\000n\000g}{chapter.19}% 20
|
||||
\BOOKMARK [0][-]{chapter.21}{\376\377\0005\000\040\000Z\000u\000s\000a\000m\000m\000e\000n\000f\000a\000s\000s\000u\000n\000g\000\040\000u\000n\000d\000\040\000A\000u\000s\000b\000l\000i\000c\000k}{}% 21
|
||||
\BOOKMARK [0][-]{appendix.22}{\376\377\000A\000\040\000A\000n\000h\000a\000n\000g}{}% 22
|
||||
\BOOKMARK [0][-]{dummy.23}{\376\377\000L\000i\000t\000e\000r\000a\000t\000u\000r\000v\000e\000r\000z\000e\000i\000c\000h\000n\000i\000s}{}% 23
|
@ -1,5 +0,0 @@
|
||||
\contentsline {chapter}{Abstrakt}{}
|
||||
\vspace {1em}\global \let \gdef empty{empty}
|
||||
\vspace {2em}
|
||||
\vspace {2em}
|
||||
\contentsline {chapter}{}{}
|
@ -1,11 +0,0 @@
|
||||
\chapter{Anhang}
|
||||
|
||||
Lorem ipsum dolor sit amet, consectetur adipiscing elit. Vivamus at pulvinar nisi. Phasellus hendrerit, diam placerat interdum iaculis, mauris justo cursus risus, in viverra purus eros at ligula. Ut metus justo, consequat a tristique posuere, laoreet nec nibh. Etiam et scelerisque mauris. Phasellus vel massa magna. Ut non neque id tortor pharetra bibendum vitae sit amet nisi. Duis nec quam quam, sed euismod justo. Pellentesque eu tellus vitae ante tempus malesuada. Nunc accumsan, quam in congue consequat, lectus lectus dapibus erat, id aliquet urna neque at massa. Nulla facilisi. Morbi ullamcorper eleifend posuere. Donec libero leo, faucibus nec bibendum at, mattis et urna. Proin consectetur, nunc ut imperdiet lobortis, magna neque tincidunt lectus, id iaculis nisi justo id nibh. Pellentesque vel sem in erat vulputate faucibus molestie ut lorem.
|
||||
|
||||
Quisque tristique urna in lorem laoreet at laoreet quam congue. Donec dolor turpis, blandit non imperdiet aliquet, blandit et felis. In lorem nisi, pretium sit amet vestibulum sed, tempus et sem. Proin non ante turpis. Nulla imperdiet fringilla convallis. Vivamus vel bibendum nisl. Pellentesque justo lectus, molestie vel luctus sed, lobortis in libero. Nulla facilisi. Aliquam erat volutpat. Suspendisse vitae nunc nunc. Sed aliquet est suscipit sapien rhoncus non adipiscing nibh consequat. Aliquam metus urna, faucibus eu vulputate non, luctus eu justo.
|
||||
|
||||
Donec urna leo, vulputate vitae porta eu, vehicula blandit libero. Phasellus eget massa et leo condimentum mollis. Nullam molestie, justo at pellentesque vulputate, sapien velit ornare diam, nec gravida lacus augue non diam. Integer mattis lacus id libero ultrices sit amet mollis neque molestie. Integer ut leo eget mi volutpat congue. Vivamus sodales, turpis id venenatis placerat, tellus purus adipiscing magna, eu aliquam nibh dolor id nibh. Pellentesque habitant morbi tristique senectus et netus et malesuada fames ac turpis egestas. Sed cursus convallis quam nec vehicula. Sed vulputate neque eget odio fringilla ac sodales urna feugiat.
|
||||
|
||||
Phasellus nisi quam, volutpat non ullamcorper eget, congue fringilla leo. Cras et erat et nibh placerat commodo id ornare est. Nulla facilisi. Aenean pulvinar scelerisque eros eget interdum. Nunc pulvinar magna ut felis varius in hendrerit dolor accumsan. Nunc pellentesque magna quis magna bibendum non laoreet erat tincidunt. Nulla facilisi.
|
||||
|
||||
Duis eget massa sem, gravida interdum ipsum. Nulla nunc nisl, hendrerit sit amet commodo vel, varius id tellus. Lorem ipsum dolor sit amet, consectetur adipiscing elit. Nunc ac dolor est. Suspendisse ultrices tincidunt metus eget accumsan. Nullam facilisis, justo vitae convallis sollicitudin, eros augue malesuada metus, nec sagittis diam nibh ut sapien. Duis blandit lectus vitae lorem aliquam nec euismod nisi volutpat. Vestibulum ornare dictum tortor, at faucibus justo tempor non. Nulla facilisi. Cras non massa nunc, eget euismod purus. Nunc metus ipsum, euismod a consectetur vel, hendrerit nec nunc.
|
@ -1,11 +0,0 @@
|
||||
\chapter{An Appendix}
|
||||
|
||||
Lorem ipsum dolor sit amet, consectetur adipiscing elit. Vivamus at pulvinar nisi. Phasellus hendrerit, diam placerat interdum iaculis, mauris justo cursus risus, in viverra purus eros at ligula. Ut metus justo, consequat a tristique posuere, laoreet nec nibh. Etiam et scelerisque mauris. Phasellus vel massa magna. Ut non neque id tortor pharetra bibendum vitae sit amet nisi. Duis nec quam quam, sed euismod justo. Pellentesque eu tellus vitae ante tempus malesuada. Nunc accumsan, quam in congue consequat, lectus lectus dapibus erat, id aliquet urna neque at massa. Nulla facilisi. Morbi ullamcorper eleifend posuere. Donec libero leo, faucibus nec bibendum at, mattis et urna. Proin consectetur, nunc ut imperdiet lobortis, magna neque tincidunt lectus, id iaculis nisi justo id nibh. Pellentesque vel sem in erat vulputate faucibus molestie ut lorem.
|
||||
|
||||
Quisque tristique urna in lorem laoreet at laoreet quam congue. Donec dolor turpis, blandit non imperdiet aliquet, blandit et felis. In lorem nisi, pretium sit amet vestibulum sed, tempus et sem. Proin non ante turpis. Nulla imperdiet fringilla convallis. Vivamus vel bibendum nisl. Pellentesque justo lectus, molestie vel luctus sed, lobortis in libero. Nulla facilisi. Aliquam erat volutpat. Suspendisse vitae nunc nunc. Sed aliquet est suscipit sapien rhoncus non adipiscing nibh consequat. Aliquam metus urna, faucibus eu vulputate non, luctus eu justo.
|
||||
|
||||
Donec urna leo, vulputate vitae porta eu, vehicula blandit libero. Phasellus eget massa et leo condimentum mollis. Nullam molestie, justo at pellentesque vulputate, sapien velit ornare diam, nec gravida lacus augue non diam. Integer mattis lacus id libero ultrices sit amet mollis neque molestie. Integer ut leo eget mi volutpat congue. Vivamus sodales, turpis id venenatis placerat, tellus purus adipiscing magna, eu aliquam nibh dolor id nibh. Pellentesque habitant morbi tristique senectus et netus et malesuada fames ac turpis egestas. Sed cursus convallis quam nec vehicula. Sed vulputate neque eget odio fringilla ac sodales urna feugiat.
|
||||
|
||||
Phasellus nisi quam, volutpat non ullamcorper eget, congue fringilla leo. Cras et erat et nibh placerat commodo id ornare est. Nulla facilisi. Aenean pulvinar scelerisque eros eget interdum. Nunc pulvinar magna ut felis varius in hendrerit dolor accumsan. Nunc pellentesque magna quis magna bibendum non laoreet erat tincidunt. Nulla facilisi.
|
||||
|
||||
Duis eget massa sem, gravida interdum ipsum. Nulla nunc nisl, hendrerit sit amet commodo vel, varius id tellus. Lorem ipsum dolor sit amet, consectetur adipiscing elit. Nunc ac dolor est. Suspendisse ultrices tincidunt metus eget accumsan. Nullam facilisis, justo vitae convallis sollicitudin, eros augue malesuada metus, nec sagittis diam nibh ut sapien. Duis blandit lectus vitae lorem aliquam nec euismod nisi volutpat. Vestibulum ornare dictum tortor, at faucibus justo tempor non. Nulla facilisi. Cras non massa nunc, eget euismod purus. Nunc metus ipsum, euismod a consectetur vel, hendrerit nec nunc.
|
@ -1,40 +0,0 @@
|
||||
@article{Reference1,
|
||||
Abstract = {We have developed an enhanced Littrow configuration extended cavity diode laser (ECDL) that can be tuned without changing the direction of the output beam. The output of a conventional Littrow ECDL is reflected from a plane mirror fixed parallel to the tuning diffraction grating. Using a free-space Michelson wavemeter to measure the laser wavelength, we can tune the laser over a range greater than 10 nm without any alteration of alignment.},
|
||||
Author = {C. J. Hawthorn and K. P. Weber and R. E. Scholten},
|
||||
Journal = {Review of Scientific Instruments},
|
||||
Month = {December},
|
||||
Number = {12},
|
||||
Numpages = {3},
|
||||
Pages = {4477--4479},
|
||||
Title = {Littrow Configuration Tunable External Cavity Diode Laser with Fixed Direction Output Beam},
|
||||
Volume = {72},
|
||||
Url = {http://link.aip.org/link/?RSI/72/4477/1},
|
||||
Year = {2001}}
|
||||
|
||||
@article{Reference3,
|
||||
Abstract = {Operating a laser diode in an extended cavity which provides frequency-selective feedback is a very effective method of reducing the laser's linewidth and improving its tunability. We have developed an extremely simple laser of this type, built from inexpensive commercial components with only a few minor modifications. A 780~nm laser built to this design has an output power of 80~mW, a linewidth of 350~kHz, and it has been continuously locked to a Doppler-free rubidium transition for several days.},
|
||||
Author = {A. S. Arnold and J. S. Wilson and M. G. Boshier},
|
||||
Journal = {Review of Scientific Instruments},
|
||||
Month = {March},
|
||||
Number = {3},
|
||||
Numpages = {4},
|
||||
Pages = {1236--1239},
|
||||
Title = {A Simple Extended-Cavity Diode Laser},
|
||||
Volume = {69},
|
||||
Url = {http://link.aip.org/link/?RSI/69/1236/1},
|
||||
Year = {1998}}
|
||||
|
||||
@article{Reference2,
|
||||
Abstract = {We present a review of the use of diode lasers in atomic physics with an extensive list of references. We discuss the relevant characteristics of diode lasers and explain how to purchase and use them. We also review the various techniques that have been used to control and narrow the spectral outputs of diode lasers. Finally we present a number of examples illustrating the use of diode lasers in atomic physics experiments. Review of Scientific Instruments is copyrighted by The American Institute of Physics.},
|
||||
Author = {Carl E. Wieman and Leo Hollberg},
|
||||
Journal = {Review of Scientific Instruments},
|
||||
Keywords = {Diode Laser},
|
||||
Month = {January},
|
||||
Number = {1},
|
||||
Numpages = {20},
|
||||
Pages = {1--20},
|
||||
Title = {Using Diode Lasers for Atomic Physics},
|
||||
Volume = {62},
|
||||
Url = {http://link.aip.org/link/?RSI/62/1/1},
|
||||
Year = {1991}}
|
||||
|