mirror of
				https://gitlab.ub.uni-bielefeld.de/sfb1288inf/nopaque.git
				synced 2025-10-31 10:42:43 +00:00 
			
		
		
		
	Merge branch 'profile-page' into development
This commit is contained in:
		| @@ -77,6 +77,9 @@ def create_app(config: Config = Config) -> Flask: | ||||
|  | ||||
|     from .main import bp as main_blueprint | ||||
|     app.register_blueprint(main_blueprint, url_prefix='/') | ||||
|      | ||||
|     from .profile import bp as profile_blueprint | ||||
|     app.register_blueprint(profile_blueprint, url_prefix='/profile') | ||||
|  | ||||
|     from .services import bp as services_blueprint | ||||
|     app.register_blueprint(services_blueprint, url_prefix='/services') | ||||
|   | ||||
| @@ -5,7 +5,7 @@ from app import db, hashids | ||||
| from app.decorators import admin_required | ||||
| from app.models import Role, User, UserSettingJobStatusMailNotificationLevel | ||||
| from app.settings.forms import ( | ||||
|     EditGeneralSettingsForm, | ||||
|     EditProfileSettingsForm, | ||||
|     EditNotificationSettingsForm | ||||
| ) | ||||
| from . import bp | ||||
| @@ -51,10 +51,10 @@ def edit_user(user_id): | ||||
|         data={'confirmed': user.confirmed, 'role': user.role.hashid}, | ||||
|         prefix='admin-edit-user-form' | ||||
|     ) | ||||
|     edit_general_settings_form = EditGeneralSettingsForm( | ||||
|     edit_profile_settings_form = EditProfileSettingsForm( | ||||
|         user, | ||||
|         data=user.to_json_serializeable(), | ||||
|         prefix='edit-general-settings-form' | ||||
|         prefix='edit-profile-settings-form' | ||||
|     ) | ||||
|     edit_notification_settings_form = EditNotificationSettingsForm( | ||||
|         data=user.to_json_serializeable(), | ||||
| @@ -68,10 +68,10 @@ def edit_user(user_id): | ||||
|         db.session.commit() | ||||
|         flash('Your changes have been saved') | ||||
|         return redirect(url_for('.edit_user', user_id=user.id)) | ||||
|     if (edit_general_settings_form.submit.data | ||||
|             and edit_general_settings_form.validate()): | ||||
|         user.email = edit_general_settings_form.email.data | ||||
|         user.username = edit_general_settings_form.username.data | ||||
|     if (edit_profile_settings_form.submit.data | ||||
|             and edit_profile_settings_form.validate()): | ||||
|         user.email = edit_profile_settings_form.email.data | ||||
|         user.username = edit_profile_settings_form.username.data | ||||
|         db.session.commit() | ||||
|         flash('Your changes have been saved') | ||||
|         return redirect(url_for('.edit_user', user_id=user.id)) | ||||
| @@ -87,7 +87,7 @@ def edit_user(user_id): | ||||
|     return render_template( | ||||
|         'admin/edit_user.html.j2', | ||||
|         admin_edit_user_form=admin_edit_user_form, | ||||
|         edit_general_settings_form=edit_general_settings_form, | ||||
|         edit_profile_settings_form=edit_profile_settings_form, | ||||
|         edit_notification_settings_form=edit_notification_settings_form, | ||||
|         title='Edit user', | ||||
|         user=user | ||||
|   | ||||
| @@ -262,6 +262,11 @@ class User(HashidMixin, UserMixin, db.Model): | ||||
|         default=UserSettingJobStatusMailNotificationLevel.END | ||||
|     ) | ||||
|     last_seen = db.Column(db.DateTime()) | ||||
|     full_name = db.Column(db.String(64)) | ||||
|     about_me = db.Column(db.String(256)) | ||||
|     location = db.Column(db.String(64)) | ||||
|     website = db.Column(db.String(128)) | ||||
|     organization = db.Column(db.String(128)) | ||||
|     # Backrefs: role: Role | ||||
|     # Relationships | ||||
|     tesseract_ocr_pipeline_models = db.relationship( | ||||
|   | ||||
							
								
								
									
										5
									
								
								app/profile/__init__.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										5
									
								
								app/profile/__init__.py
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,5 @@ | ||||
| from flask import Blueprint | ||||
|  | ||||
|  | ||||
| bp = Blueprint('profile', __name__) | ||||
| from . import routes  # noqa | ||||
							
								
								
									
										87
									
								
								app/profile/forms.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										87
									
								
								app/profile/forms.py
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,87 @@ | ||||
| from flask_wtf import FlaskForm | ||||
| from wtforms import ( | ||||
|     FileField, | ||||
|     StringField, | ||||
|     SubmitField, | ||||
|     TextAreaField, | ||||
|     ValidationError | ||||
| ) | ||||
| from wtforms.validators import ( | ||||
|     InputRequired, | ||||
|     Email, | ||||
|     Length, | ||||
|     Regexp | ||||
| ) | ||||
| from app.models import User | ||||
| from app.auth import USERNAME_REGEX | ||||
|  | ||||
| class EditProfileSettingsForm(FlaskForm): | ||||
|     user_avatar = FileField( | ||||
|         'Image File' | ||||
|     ) | ||||
|     email = StringField( | ||||
|         'E-Mail', | ||||
|         validators=[InputRequired(), Length(max=254), Email()] | ||||
|     ) | ||||
|     username = StringField( | ||||
|         'Username', | ||||
|         validators=[ | ||||
|             InputRequired(), | ||||
|             Length(max=64), | ||||
|             Regexp( | ||||
|                 USERNAME_REGEX, | ||||
|                 message=( | ||||
|                     'Usernames must have only letters, numbers, dots or ' | ||||
|                     'underscores' | ||||
|                 ) | ||||
|             ) | ||||
|         ] | ||||
|     ) | ||||
|     full_name = StringField( | ||||
|         'Full name', | ||||
|         validators=[Length(max=128)] | ||||
|     ) | ||||
|     about_me = TextAreaField( | ||||
|         'About me',  | ||||
|         validators=[ | ||||
|             Length(max=254) | ||||
|         ] | ||||
|     ) | ||||
|     website = StringField( | ||||
|         'Website', | ||||
|         validators=[ | ||||
|             Length(max=254) | ||||
|         ] | ||||
|     ) | ||||
|     organization = StringField( | ||||
|         'Organization', | ||||
|         validators=[ | ||||
|             Length(max=128) | ||||
|         ] | ||||
|     ) | ||||
|     location = StringField( | ||||
|         'Location', | ||||
|         validators=[ | ||||
|             Length(max=128) | ||||
|         ] | ||||
|     ) | ||||
|  | ||||
|     submit = SubmitField() | ||||
|  | ||||
|     def __init__(self, user, *args, **kwargs): | ||||
|         super().__init__(*args, **kwargs) | ||||
|         self.user = user | ||||
|  | ||||
|     def validate_email(self, field): | ||||
|         if (field.data != self.user.email | ||||
|                 and User.query.filter_by(email=field.data).first()): | ||||
|             raise ValidationError('Email already registered') | ||||
|  | ||||
|     def validate_username(self, field): | ||||
|         if (field.data != self.user.username | ||||
|                 and User.query.filter_by(username=field.data).first()): | ||||
|             raise ValidationError('Username already in use') | ||||
|     | ||||
|     def validate_image_file(self, field): | ||||
|         if not field.data.filename.lower().endswith('.jpg' or '.png' or '.jpeg'): | ||||
|             raise ValidationError('only .jpg, .png and .jpeg!') | ||||
							
								
								
									
										61
									
								
								app/profile/routes.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										61
									
								
								app/profile/routes.py
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,61 @@ | ||||
| from flask import flash, redirect, render_template, url_for | ||||
| from flask_login import current_user, login_required | ||||
| from app import db | ||||
| from app.models import User | ||||
| from . import bp | ||||
| from .forms import ( | ||||
|   EditProfileSettingsForm | ||||
| ) | ||||
|  | ||||
| @bp.route('') | ||||
| @login_required | ||||
| def profile(): | ||||
|   user_image = 'static/images/user_avatar.png' | ||||
|   user_name = current_user.username | ||||
|   last_seen = f'{current_user.last_seen.strftime("%Y-%m-%d %H:%M")}' | ||||
|   location = 'Bielefeld' | ||||
|   about_me = '''Lorem ipsum dolor sit amet, consetetur sadipscing elitr,  | ||||
|                 sed diam nonumy eirmod tempor invidunt ut labore et dolore  | ||||
|                 magna aliquyam erat, sed diam voluptua. At vero eos et accusam  | ||||
|                 et justo duo dolores et ea rebum. Stet clita kasd gubergren,  | ||||
|                 no sea takimat''' | ||||
|   full_name = 'Inga Kirschnick' | ||||
|   email = current_user.email | ||||
|   website = 'https://nopaque.uni-bielefeld.de' | ||||
|   organization = 'Universität Bielefeld' | ||||
|   member_since = f'{current_user.member_since.strftime("%Y-%m-%d")}' | ||||
|   return render_template('profile/profile_page.html.j2',  | ||||
|   user_image=user_image,  | ||||
|   user_name=user_name,  | ||||
|   last_seen=last_seen, | ||||
|   location=location, | ||||
|   about_me=about_me, | ||||
|   full_name=full_name, | ||||
|   email=email, | ||||
|   website=website, | ||||
|   organization=organization, | ||||
|   member_since=member_since) | ||||
|  | ||||
| @bp.route('/edit') | ||||
| @login_required | ||||
| def edit_profile(): | ||||
|   edit_profile_settings_form = EditProfileSettingsForm( | ||||
|         current_user, | ||||
|         data=current_user.to_json_serializeable(), | ||||
|         prefix='edit-profile-settings-form' | ||||
|   ) | ||||
|   if (edit_profile_settings_form.submit.data | ||||
|             and edit_profile_settings_form.validate()): | ||||
|         current_user.email = edit_profile_settings_form.email.data | ||||
|         current_user.username = edit_profile_settings_form.username.data | ||||
|         current_user.about_me = edit_profile_settings_form.about_me.data | ||||
|         current_user.location = edit_profile_settings_form.location.data | ||||
|         current_user.organization = edit_profile_settings_form.organization.data | ||||
|         current_user.website = edit_profile_settings_form.website.data | ||||
|         current_user.full_name = edit_profile_settings_form.full_name.data | ||||
|         db.session.commit() | ||||
|         flash('Your changes have been saved') | ||||
|         return redirect(url_for('.profile.edit_profile')) | ||||
|   return render_template('profile/edit_profile.html.j2', | ||||
|   edit_profile_settings_form=edit_profile_settings_form, | ||||
|   title='Edit Profile') | ||||
| @@ -1,10 +1,12 @@ | ||||
| from flask_wtf import FlaskForm | ||||
| from wtforms import ( | ||||
|     BooleanField, | ||||
|     FileField, | ||||
|     PasswordField, | ||||
|     SelectField, | ||||
|     StringField, | ||||
|     SubmitField, | ||||
|     TextAreaField, | ||||
|     ValidationError | ||||
| ) | ||||
| from wtforms.validators import ( | ||||
| @@ -46,7 +48,10 @@ class ChangePasswordForm(FlaskForm): | ||||
|             raise ValidationError('Invalid password') | ||||
|  | ||||
|  | ||||
| class EditGeneralSettingsForm(FlaskForm): | ||||
| class EditProfileSettingsForm(FlaskForm): | ||||
|     user_avatar = FileField( | ||||
|         'Image File' | ||||
|     ) | ||||
|     email = StringField( | ||||
|         'E-Mail', | ||||
|         validators=[InputRequired(), Length(max=254), Email()] | ||||
| @@ -65,6 +70,35 @@ class EditGeneralSettingsForm(FlaskForm): | ||||
|             ) | ||||
|         ] | ||||
|     ) | ||||
|     full_name = StringField( | ||||
|         'Full name', | ||||
|         validators=[Length(max=128)] | ||||
|     ) | ||||
|     about_me = TextAreaField( | ||||
|         'About me',  | ||||
|         validators=[ | ||||
|             Length(max=254) | ||||
|         ] | ||||
|     ) | ||||
|     website = StringField( | ||||
|         'Website', | ||||
|         validators=[ | ||||
|             Length(max=254) | ||||
|         ] | ||||
|     ) | ||||
|     organization = StringField( | ||||
|         'Organization', | ||||
|         validators=[ | ||||
|             Length(max=128) | ||||
|         ] | ||||
|     ) | ||||
|     location = StringField( | ||||
|         'Location', | ||||
|         validators=[ | ||||
|             Length(max=128) | ||||
|         ] | ||||
|     ) | ||||
|  | ||||
|     submit = SubmitField() | ||||
|  | ||||
|     def __init__(self, user, *args, **kwargs): | ||||
| @@ -80,6 +114,10 @@ class EditGeneralSettingsForm(FlaskForm): | ||||
|         if (field.data != self.user.username | ||||
|                 and User.query.filter_by(username=field.data).first()): | ||||
|             raise ValidationError('Username already in use') | ||||
|     | ||||
|     def validate_image_file(self, field): | ||||
|         if not field.data.filename.lower().endswith('.jpg' or '.png' or '.jpeg'): | ||||
|             raise ValidationError('only .jpg, .png and .jpeg!') | ||||
|  | ||||
|  | ||||
| class EditNotificationSettingsForm(FlaskForm): | ||||
| @@ -96,3 +134,9 @@ class EditNotificationSettingsForm(FlaskForm): | ||||
|             (x.name, x.name.capitalize()) | ||||
|             for x in UserSettingJobStatusMailNotificationLevel | ||||
|         ] | ||||
|  | ||||
| class EditPrivacySettingsForm(FlaskForm): | ||||
|     public_profile = BooleanField( | ||||
|         'Public profile' | ||||
|     ) | ||||
|     submit = SubmitField() | ||||
|   | ||||
| @@ -5,8 +5,8 @@ from app.models import UserSettingJobStatusMailNotificationLevel | ||||
| from . import bp | ||||
| from .forms import ( | ||||
|     ChangePasswordForm, | ||||
|     EditGeneralSettingsForm, | ||||
|     EditNotificationSettingsForm | ||||
|     EditNotificationSettingsForm, | ||||
|     EditPrivacySettingsForm | ||||
| ) | ||||
|  | ||||
|  | ||||
| @@ -17,28 +17,20 @@ def settings(): | ||||
|         current_user, | ||||
|         prefix='change-password-form' | ||||
|     ) | ||||
|     edit_general_settings_form = EditGeneralSettingsForm( | ||||
|         current_user, | ||||
|         data=current_user.to_json_serializeable(), | ||||
|         prefix='edit-general-settings-form' | ||||
|     ) | ||||
|     edit_notification_settings_form = EditNotificationSettingsForm( | ||||
|         data=current_user.to_json_serializeable(), | ||||
|         prefix='edit-notification-settings-form' | ||||
|     ) | ||||
|  | ||||
|     edit_privacy_settings_form = EditPrivacySettingsForm( | ||||
|         data=current_user.to_json_serializeable(), | ||||
|         prefix='edit-privacy-settings-form' | ||||
|     ) | ||||
|     if change_password_form.submit.data and change_password_form.validate(): | ||||
|         current_user.password = change_password_form.new_password.data | ||||
|         db.session.commit() | ||||
|         flash('Your changes have been saved') | ||||
|         return redirect(url_for('.index')) | ||||
|     if (edit_general_settings_form.submit.data | ||||
|             and edit_general_settings_form.validate()): | ||||
|         current_user.email = edit_general_settings_form.email.data | ||||
|         current_user.username = edit_general_settings_form.username.data | ||||
|         db.session.commit() | ||||
|         flash('Your changes have been saved') | ||||
|         return redirect(url_for('.settings')) | ||||
|      | ||||
|     if (edit_notification_settings_form.submit.data | ||||
|             and edit_notification_settings_form.validate()): | ||||
|         current_user.setting_job_status_mail_notification_level = ( | ||||
| @@ -52,7 +44,7 @@ def settings(): | ||||
|     return render_template( | ||||
|         'settings/settings.html.j2', | ||||
|         change_password_form=change_password_form, | ||||
|         edit_general_settings_form=edit_general_settings_form, | ||||
|         edit_notification_settings_form=edit_notification_settings_form, | ||||
|         edit_privacy_settings_form=edit_privacy_settings_form, | ||||
|         title='Settings' | ||||
|     ) | ||||
|   | ||||
							
								
								
									
										
											BIN
										
									
								
								app/static/images/user_avatar.png
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										
											BIN
										
									
								
								app/static/images/user_avatar.png
									
									
									
									
									
										Normal file
									
								
							
										
											Binary file not shown.
										
									
								
							| After Width: | Height: | Size: 2.0 KiB | 
| @@ -2,8 +2,16 @@ | ||||
|   <li> | ||||
|     <div class="user-view"> | ||||
|       <div class="background primary-color"></div> | ||||
|       <span class="white-text name">{{ current_user.username }}</span> | ||||
|       <span class="white-text email">{{ current_user.email }}</span> | ||||
|       <div class="row"> | ||||
|         <div class="col s2"> | ||||
|           <a href="{{ url_for('profile.profile') }}"> | ||||
|           <i class="material-icons" style="color:white; font-size:3em; margin-top: 25px; margin-left:-15px;">account_circle</i></div> | ||||
|           </a> | ||||
|         <div class="col s10"> | ||||
|           <span class="white-text name">{{ current_user.username }}</span> | ||||
|           <span class="white-text email">{{ current_user.email }}</span> | ||||
|         </div> | ||||
|       </div> | ||||
|     </div> | ||||
|   </li> | ||||
|   <li><a href="{{ url_for('main.index') }}">nopaque</a></li> | ||||
|   | ||||
							
								
								
									
										43
									
								
								app/templates/profile/edit_profile.html.j2
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										43
									
								
								app/templates/profile/edit_profile.html.j2
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,43 @@ | ||||
| {% extends "base.html.j2" %} | ||||
| {% import "materialize/wtf.html.j2" as wtf %} | ||||
|  | ||||
| {% block page_content %} | ||||
| <div class="container"> | ||||
|   <div class="row"> | ||||
|     <div class="col s12"> | ||||
|         <h1 id="title">{{ title }}</h1> | ||||
|       </div> | ||||
|     <div class="col s12"> | ||||
|       <form method="POST"> | ||||
|         {{ edit_profile_settings_form.hidden_tag() }} | ||||
|         <div class="card"> | ||||
|           <div class="card-content"> | ||||
|             <div class="row"> | ||||
|               <div class="col s1"></div> | ||||
|               <div class="col s3"> | ||||
|                 <img src="{{ url_for('static', filename='images/user_avatar.png') }}" alt="user-image" class="circle responsive-img"> | ||||
|                 {{wtf.render_field(edit_profile_settings_form.user_avatar, accept='image/*', class='file-path validate')}} | ||||
|               </div> | ||||
|               <div class="col s1"></div> | ||||
|               <div class="col s7"> | ||||
|                 {{ wtf.render_field(edit_profile_settings_form.username, material_icon='person') }} | ||||
|                 {{ wtf.render_field(edit_profile_settings_form.email, material_icon='email') }} | ||||
|                 {{ wtf.render_field(edit_profile_settings_form.full_name, material_icon='badge') }} | ||||
|                 {{ wtf.render_field(edit_profile_settings_form.about_me, material_icon='description') }} | ||||
|                 {{ wtf.render_field(edit_profile_settings_form.website, material_icon='laptop') }} | ||||
|                 {{ wtf.render_field(edit_profile_settings_form.organization, material_icon='business') }} | ||||
|                 {{ wtf.render_field(edit_profile_settings_form.location, material_icon='location_on') }} | ||||
|               </div> | ||||
|             </div> | ||||
|           </div> | ||||
|           <div class="card-action"> | ||||
|             <div class="right-align"> | ||||
|               {{ wtf.render_field(edit_profile_settings_form.submit, material_icon='send') }} | ||||
|             </div> | ||||
|           </div> | ||||
|         </div> | ||||
|       </form> | ||||
|     </div> | ||||
|   </div> | ||||
| </div> | ||||
| {% endblock page_content %} | ||||
							
								
								
									
										85
									
								
								app/templates/profile/profile_page.html.j2
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										85
									
								
								app/templates/profile/profile_page.html.j2
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,85 @@ | ||||
| {% extends "base.html.j2" %} | ||||
| {% import "materialize/wtf.html.j2" as wtf %} | ||||
|  | ||||
| {% block page_content %} | ||||
|     <div class="container"> | ||||
|         <div class="row"> | ||||
|           <div class="col s12"> | ||||
|             <div class="card"> | ||||
|               <div class="card-content"> | ||||
|                 <div class="row"> | ||||
|                   <div class="col s1"></div> | ||||
|                   <div class="col s3"> | ||||
|                     {% if about_me %} | ||||
|                     <img src="{{ user_image }}" alt="user-image" class="circle responsive-img" style="margin-top:50px;"> | ||||
|                     {% else %} | ||||
|                     <img src="{{ user_image }}" alt="user-image" class="circle responsive-img"> | ||||
|                     {% endif %} | ||||
|                   </div> | ||||
|                   <div class="col s1"></div> | ||||
|                   <div class="col s7"> | ||||
|                     <h3>{{ user_name }}</h3> | ||||
|                     <div class="chip">Last seen: {{ last_seen }}</div> | ||||
|                     {% if location %} | ||||
|                     <p><span class="material-icons" style="margin-right:20px; margin-top:20px;">location_on</span><i>{{ location }}</i></p> | ||||
|                     {% endif %} | ||||
|                     <p></p> | ||||
|                     <br> | ||||
|                     {% if about_me%} | ||||
|                     <div class="card"> | ||||
|                       <div class="card-content"> | ||||
|                         <span class="card-title">About me</span> | ||||
|                         <p>{{ about_me }}</p> | ||||
|                       </div> | ||||
|                     </div> | ||||
|                     {% endif %} | ||||
|                   </div> | ||||
|                 </div> | ||||
|                 <div class="row"> | ||||
|                   <div class="col s1"></div> | ||||
|                   <div class="col s6"> | ||||
|                     <table> | ||||
|                       {% if full_name %} | ||||
|                       <tr> | ||||
|                         <td><span class="material-icons">person</span></td> | ||||
|                         <td>{{ full_name }} </td> | ||||
|                       </tr> | ||||
|                       {% endif %} | ||||
|                       {% if email %} | ||||
|                       <tr> | ||||
|                         <td><span class="material-icons">email</span></td> | ||||
|                         <td>{{ email }}</td> | ||||
|                       </tr> | ||||
|                       {% endif %} | ||||
|                       {% if website %} | ||||
|                       <tr> | ||||
|                         <td><span class="material-icons">laptop</span></td> | ||||
|                         <td><a href="{{ website }}">{{ website }}</a></td> | ||||
|                       </tr> | ||||
|                       {% endif %} | ||||
|                       {% if organization %} | ||||
|                       <tr> | ||||
|                         <td><span class="material-icons">business</span></td> | ||||
|                         <td>{{ organization }}</td> | ||||
|                       </tr> | ||||
|                       {% endif %} | ||||
|                     </table> | ||||
|                     <br> | ||||
|                     <p><i>Member since: {{ member_since }}</i></p> | ||||
|                     <p></p> | ||||
|                     <br> | ||||
|                     <a class="waves-effect waves-light btn-small" href="{{ url_for('settings.settings') }}">Edit profile</a> | ||||
|                   </div> | ||||
|                 </div> | ||||
|               </div> | ||||
|             </div> | ||||
|           </div> | ||||
|         </div> | ||||
|         <div class="row"> | ||||
|           <div class="col s6"> | ||||
|            | ||||
|           </div> | ||||
|  | ||||
|         </div> | ||||
|     </div> | ||||
| {% endblock page_content %} | ||||
| @@ -8,31 +8,26 @@ | ||||
|     <div class="col s12"> | ||||
|       <h1 id="title">{{ title }}</h1> | ||||
|     </div> | ||||
|  | ||||
|     <div class="col s12"> | ||||
|       <form method="POST"> | ||||
|         {{ edit_general_settings_form.hidden_tag() }} | ||||
|         {{ edit_notification_settings_form.hidden_tag() }} | ||||
|         <div class="card"> | ||||
|           <div class="card-content"> | ||||
|             <span class="card-title">General settings</span> | ||||
|             {{ wtf.render_field(edit_general_settings_form.username, material_icon='person') }} | ||||
|             {{ wtf.render_field(edit_general_settings_form.email, material_icon='email') }} | ||||
|             <span class="card-title">Notification settings</span> | ||||
|             {{ wtf.render_field(edit_notification_settings_form.job_status_mail_notification_level, material_icon='notifications') }} | ||||
|           </div> | ||||
|           <div class="card-action"> | ||||
|             <div class="right-align"> | ||||
|               {{ wtf.render_field(edit_general_settings_form.submit, material_icon='send') }} | ||||
|               {{ wtf.render_field(edit_notification_settings_form.submit, material_icon='send') }} | ||||
|             </div> | ||||
|           </div> | ||||
|         </form> | ||||
|       </div> | ||||
|     </form> | ||||
|  | ||||
|     <form method="POST"> | ||||
|       {{ edit_notification_settings_form.hidden_tag() }} | ||||
|         </div> | ||||
|       </form> | ||||
|        | ||||
|       <div class="card"> | ||||
|         <div class="card-content"> | ||||
|           <span class="card-title">Notification settings</span> | ||||
|           {{ wtf.render_field(edit_notification_settings_form.job_status_mail_notification_level, material_icon='notifications') }} | ||||
|           <span class="card-title">Privacy settings</span> | ||||
|           {{ wtf.render_field(edit_privacy_settings_form.public_profile) }} | ||||
|         </div> | ||||
|         <div class="card-action"> | ||||
|           <div class="right-align"> | ||||
| @@ -40,36 +35,36 @@ | ||||
|           </div> | ||||
|         </div> | ||||
|       </div> | ||||
|     </form> | ||||
|  | ||||
|     <form method="POST"> | ||||
|       {{ change_password_form.hidden_tag() }} | ||||
|       <div class="card"> | ||||
|         <div class="card-content"> | ||||
|           <span class="card-title">Change Password</span> | ||||
|           {{ wtf.render_field(change_password_form.password, material_icon='vpn_key') }} | ||||
|           {{ wtf.render_field(change_password_form.new_password, material_icon='vpn_key') }} | ||||
|           {{ wtf.render_field(change_password_form.new_password_2, material_icon='vpn_key') }} | ||||
|         </div> | ||||
|         <div class="card-action"> | ||||
|           <div class="right-align"> | ||||
|             {{ wtf.render_field(change_password_form.submit, material_icon='send') }} | ||||
|       <form method="POST"> | ||||
|         {{ change_password_form.hidden_tag() }} | ||||
|         <div class="card"> | ||||
|           <div class="card-content"> | ||||
|             <span class="card-title">Change Password</span> | ||||
|             {{ wtf.render_field(change_password_form.password, material_icon='vpn_key') }} | ||||
|             {{ wtf.render_field(change_password_form.new_password, material_icon='vpn_key') }} | ||||
|             {{ wtf.render_field(change_password_form.new_password_2, material_icon='vpn_key') }} | ||||
|           </div> | ||||
|           <div class="card-action"> | ||||
|             <div class="right-align"> | ||||
|               {{ wtf.render_field(change_password_form.submit, material_icon='send') }} | ||||
|             </div> | ||||
|           </div> | ||||
|         </div> | ||||
|       </div> | ||||
|     </form> | ||||
|       </form> | ||||
|  | ||||
|     <div class="card"> | ||||
|       <div class="card-content"> | ||||
|         <span class="card-title">Delete account</span> | ||||
|         <p>Deleting an account has the following effects:</p> | ||||
|         <ul> | ||||
|           <li>All data associated with your corpora and jobs will be permanently deleted.</li> | ||||
|           <li>All settings will be permanently deleted.</li> | ||||
|         </ul> | ||||
|       </div> | ||||
|       <div class="card-action right-align"> | ||||
|         <a class="btn red waves-effect waves-light" id="delete-user"><i class="material-icons left">delete</i>Delete</a> | ||||
|       <div class="card"> | ||||
|         <div class="card-content"> | ||||
|           <span class="card-title">Delete account</span> | ||||
|           <p>Deleting an account has the following effects:</p> | ||||
|           <ul> | ||||
|             <li>All data associated with your corpora and jobs will be permanently deleted.</li> | ||||
|             <li>All settings will be permanently deleted.</li> | ||||
|           </ul> | ||||
|         </div> | ||||
|         <div class="card-action right-align"> | ||||
|           <a class="btn red waves-effect waves-light" id="delete-user"><i class="material-icons left">delete</i>Delete</a> | ||||
|         </div> | ||||
|       </div> | ||||
|     </div> | ||||
|   </div> | ||||
|   | ||||
							
								
								
									
										44
									
								
								migrations/versions/4820fa2e3ee2_.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										44
									
								
								migrations/versions/4820fa2e3ee2_.py
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,44 @@ | ||||
| """Add profile pages columns to users table | ||||
|  | ||||
| Revision ID: 4820fa2e3ee2 | ||||
| Revises: 31b9c0259e6b | ||||
| Create Date: 2022-11-30 10:03:49.080576 | ||||
|  | ||||
| """ | ||||
| from alembic import op | ||||
| import sqlalchemy as sa | ||||
|  | ||||
| revision = '4820fa2e3ee2' | ||||
| down_revision = '31b9c0259e6b' | ||||
| branch_labels = None | ||||
| depends_on = None | ||||
|  | ||||
| def upgrade(): | ||||
|  | ||||
|     op.add_column( | ||||
|         'users', | ||||
|         sa.Column('full_name', sa.String(length=64), nullable=True) | ||||
|     ) | ||||
|     op.add_column( | ||||
|         'users', | ||||
|         sa.Column('about_me', sa.String(length=256), nullable=True) | ||||
|     ) | ||||
|     op.add_column( | ||||
|         'users', | ||||
|         sa.Column('location', sa.String(length=64), nullable=True) | ||||
|     ) | ||||
|     op.add_column( | ||||
|         'users', | ||||
|         sa.Column('website', sa.String(length=128), nullable=True) | ||||
|     ) | ||||
|     op.add_column( | ||||
|         'users', | ||||
|         sa.Column('organization', sa.String(length=128), nullable=True) | ||||
|     ) | ||||
|  | ||||
| def downgrade(): | ||||
|     op.drop_column('users', 'organization') | ||||
|     op.drop_column('users', 'website') | ||||
|     op.drop_column('users', 'location') | ||||
|     op.drop_column('users', 'about_me') | ||||
|     op.drop_column('users', 'full_name') | ||||
		Reference in New Issue
	
	Block a user