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:
		| @@ -78,6 +78,9 @@ def create_app(config: Config = Config) -> Flask: | |||||||
|     from .main import bp as main_blueprint |     from .main import bp as main_blueprint | ||||||
|     app.register_blueprint(main_blueprint, url_prefix='/') |     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 |     from .services import bp as services_blueprint | ||||||
|     app.register_blueprint(services_blueprint, url_prefix='/services') |     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.decorators import admin_required | ||||||
| from app.models import Role, User, UserSettingJobStatusMailNotificationLevel | from app.models import Role, User, UserSettingJobStatusMailNotificationLevel | ||||||
| from app.settings.forms import ( | from app.settings.forms import ( | ||||||
|     EditGeneralSettingsForm, |     EditProfileSettingsForm, | ||||||
|     EditNotificationSettingsForm |     EditNotificationSettingsForm | ||||||
| ) | ) | ||||||
| from . import bp | from . import bp | ||||||
| @@ -51,10 +51,10 @@ def edit_user(user_id): | |||||||
|         data={'confirmed': user.confirmed, 'role': user.role.hashid}, |         data={'confirmed': user.confirmed, 'role': user.role.hashid}, | ||||||
|         prefix='admin-edit-user-form' |         prefix='admin-edit-user-form' | ||||||
|     ) |     ) | ||||||
|     edit_general_settings_form = EditGeneralSettingsForm( |     edit_profile_settings_form = EditProfileSettingsForm( | ||||||
|         user, |         user, | ||||||
|         data=user.to_json_serializeable(), |         data=user.to_json_serializeable(), | ||||||
|         prefix='edit-general-settings-form' |         prefix='edit-profile-settings-form' | ||||||
|     ) |     ) | ||||||
|     edit_notification_settings_form = EditNotificationSettingsForm( |     edit_notification_settings_form = EditNotificationSettingsForm( | ||||||
|         data=user.to_json_serializeable(), |         data=user.to_json_serializeable(), | ||||||
| @@ -68,10 +68,10 @@ def edit_user(user_id): | |||||||
|         db.session.commit() |         db.session.commit() | ||||||
|         flash('Your changes have been saved') |         flash('Your changes have been saved') | ||||||
|         return redirect(url_for('.edit_user', user_id=user.id)) |         return redirect(url_for('.edit_user', user_id=user.id)) | ||||||
|     if (edit_general_settings_form.submit.data |     if (edit_profile_settings_form.submit.data | ||||||
|             and edit_general_settings_form.validate()): |             and edit_profile_settings_form.validate()): | ||||||
|         user.email = edit_general_settings_form.email.data |         user.email = edit_profile_settings_form.email.data | ||||||
|         user.username = edit_general_settings_form.username.data |         user.username = edit_profile_settings_form.username.data | ||||||
|         db.session.commit() |         db.session.commit() | ||||||
|         flash('Your changes have been saved') |         flash('Your changes have been saved') | ||||||
|         return redirect(url_for('.edit_user', user_id=user.id)) |         return redirect(url_for('.edit_user', user_id=user.id)) | ||||||
| @@ -87,7 +87,7 @@ def edit_user(user_id): | |||||||
|     return render_template( |     return render_template( | ||||||
|         'admin/edit_user.html.j2', |         'admin/edit_user.html.j2', | ||||||
|         admin_edit_user_form=admin_edit_user_form, |         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, |         edit_notification_settings_form=edit_notification_settings_form, | ||||||
|         title='Edit user', |         title='Edit user', | ||||||
|         user=user |         user=user | ||||||
|   | |||||||
| @@ -262,6 +262,11 @@ class User(HashidMixin, UserMixin, db.Model): | |||||||
|         default=UserSettingJobStatusMailNotificationLevel.END |         default=UserSettingJobStatusMailNotificationLevel.END | ||||||
|     ) |     ) | ||||||
|     last_seen = db.Column(db.DateTime()) |     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 |     # Backrefs: role: Role | ||||||
|     # Relationships |     # Relationships | ||||||
|     tesseract_ocr_pipeline_models = db.relationship( |     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 flask_wtf import FlaskForm | ||||||
| from wtforms import ( | from wtforms import ( | ||||||
|     BooleanField, |     BooleanField, | ||||||
|  |     FileField, | ||||||
|     PasswordField, |     PasswordField, | ||||||
|     SelectField, |     SelectField, | ||||||
|     StringField, |     StringField, | ||||||
|     SubmitField, |     SubmitField, | ||||||
|  |     TextAreaField, | ||||||
|     ValidationError |     ValidationError | ||||||
| ) | ) | ||||||
| from wtforms.validators import ( | from wtforms.validators import ( | ||||||
| @@ -46,7 +48,10 @@ class ChangePasswordForm(FlaskForm): | |||||||
|             raise ValidationError('Invalid password') |             raise ValidationError('Invalid password') | ||||||
|  |  | ||||||
|  |  | ||||||
| class EditGeneralSettingsForm(FlaskForm): | class EditProfileSettingsForm(FlaskForm): | ||||||
|  |     user_avatar = FileField( | ||||||
|  |         'Image File' | ||||||
|  |     ) | ||||||
|     email = StringField( |     email = StringField( | ||||||
|         'E-Mail', |         'E-Mail', | ||||||
|         validators=[InputRequired(), Length(max=254), Email()] |         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() |     submit = SubmitField() | ||||||
|  |  | ||||||
|     def __init__(self, user, *args, **kwargs): |     def __init__(self, user, *args, **kwargs): | ||||||
| @@ -81,6 +115,10 @@ class EditGeneralSettingsForm(FlaskForm): | |||||||
|                 and User.query.filter_by(username=field.data).first()): |                 and User.query.filter_by(username=field.data).first()): | ||||||
|             raise ValidationError('Username already in use') |             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): | class EditNotificationSettingsForm(FlaskForm): | ||||||
|     job_status_mail_notification_level = SelectField( |     job_status_mail_notification_level = SelectField( | ||||||
| @@ -96,3 +134,9 @@ class EditNotificationSettingsForm(FlaskForm): | |||||||
|             (x.name, x.name.capitalize()) |             (x.name, x.name.capitalize()) | ||||||
|             for x in UserSettingJobStatusMailNotificationLevel |             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 . import bp | ||||||
| from .forms import ( | from .forms import ( | ||||||
|     ChangePasswordForm, |     ChangePasswordForm, | ||||||
|     EditGeneralSettingsForm, |     EditNotificationSettingsForm, | ||||||
|     EditNotificationSettingsForm |     EditPrivacySettingsForm | ||||||
| ) | ) | ||||||
|  |  | ||||||
|  |  | ||||||
| @@ -17,28 +17,20 @@ def settings(): | |||||||
|         current_user, |         current_user, | ||||||
|         prefix='change-password-form' |         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( |     edit_notification_settings_form = EditNotificationSettingsForm( | ||||||
|         data=current_user.to_json_serializeable(), |         data=current_user.to_json_serializeable(), | ||||||
|         prefix='edit-notification-settings-form' |         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(): |     if change_password_form.submit.data and change_password_form.validate(): | ||||||
|         current_user.password = change_password_form.new_password.data |         current_user.password = change_password_form.new_password.data | ||||||
|         db.session.commit() |         db.session.commit() | ||||||
|         flash('Your changes have been saved') |         flash('Your changes have been saved') | ||||||
|         return redirect(url_for('.index')) |         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 |     if (edit_notification_settings_form.submit.data | ||||||
|             and edit_notification_settings_form.validate()): |             and edit_notification_settings_form.validate()): | ||||||
|         current_user.setting_job_status_mail_notification_level = ( |         current_user.setting_job_status_mail_notification_level = ( | ||||||
| @@ -52,7 +44,7 @@ def settings(): | |||||||
|     return render_template( |     return render_template( | ||||||
|         'settings/settings.html.j2', |         'settings/settings.html.j2', | ||||||
|         change_password_form=change_password_form, |         change_password_form=change_password_form, | ||||||
|         edit_general_settings_form=edit_general_settings_form, |  | ||||||
|         edit_notification_settings_form=edit_notification_settings_form, |         edit_notification_settings_form=edit_notification_settings_form, | ||||||
|  |         edit_privacy_settings_form=edit_privacy_settings_form, | ||||||
|         title='Settings' |         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,9 +2,17 @@ | |||||||
|   <li> |   <li> | ||||||
|     <div class="user-view"> |     <div class="user-view"> | ||||||
|       <div class="background primary-color"></div> |       <div class="background primary-color"></div> | ||||||
|  |       <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 name">{{ current_user.username }}</span> | ||||||
|           <span class="white-text email">{{ current_user.email }}</span> |           <span class="white-text email">{{ current_user.email }}</span> | ||||||
|         </div> |         </div> | ||||||
|  |       </div> | ||||||
|  |     </div> | ||||||
|   </li> |   </li> | ||||||
|   <li><a href="{{ url_for('main.index') }}">nopaque</a></li> |   <li><a href="{{ url_for('main.index') }}">nopaque</a></li> | ||||||
|   <li><a href="{{ url_for('main.news') }}"><i class="material-icons left">email</i>News</a></li> |   <li><a href="{{ url_for('main.news') }}"><i class="material-icons left">email</i>News</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,25 +8,7 @@ | |||||||
|     <div class="col s12"> |     <div class="col s12"> | ||||||
|       <h1 id="title">{{ title }}</h1> |       <h1 id="title">{{ title }}</h1> | ||||||
|     </div> |     </div> | ||||||
|  |  | ||||||
|     <div class="col s12"> |     <div class="col s12"> | ||||||
|       <form method="POST"> |  | ||||||
|         {{ edit_general_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') }} |  | ||||||
|           </div> |  | ||||||
|           <div class="card-action"> |  | ||||||
|             <div class="right-align"> |  | ||||||
|               {{ wtf.render_field(edit_general_settings_form.submit, material_icon='send') }} |  | ||||||
|             </div> |  | ||||||
|           </div> |  | ||||||
|         </form> |  | ||||||
|       </div> |  | ||||||
|     </form> |  | ||||||
|  |  | ||||||
|       <form method="POST"> |       <form method="POST"> | ||||||
|         {{ edit_notification_settings_form.hidden_tag() }} |         {{ edit_notification_settings_form.hidden_tag() }} | ||||||
|         <div class="card"> |         <div class="card"> | ||||||
| @@ -42,6 +24,18 @@ | |||||||
|         </div> |         </div> | ||||||
|       </form> |       </form> | ||||||
|        |        | ||||||
|  |       <div class="card"> | ||||||
|  |         <div class="card-content"> | ||||||
|  |           <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"> | ||||||
|  |             {{ wtf.render_field(edit_notification_settings_form.submit, material_icon='send') }} | ||||||
|  |           </div> | ||||||
|  |         </div> | ||||||
|  |       </div> | ||||||
|  |  | ||||||
|       <form method="POST"> |       <form method="POST"> | ||||||
|         {{ change_password_form.hidden_tag() }} |         {{ change_password_form.hidden_tag() }} | ||||||
|         <div class="card"> |         <div class="card"> | ||||||
| @@ -73,6 +67,7 @@ | |||||||
|         </div> |         </div> | ||||||
|       </div> |       </div> | ||||||
|     </div> |     </div> | ||||||
|  |   </div> | ||||||
| </div> | </div> | ||||||
| {% endblock page_content %} | {% endblock page_content %} | ||||||
|  |  | ||||||
|   | |||||||
							
								
								
									
										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