Compare commits

..

4 Commits

Author SHA1 Message Date
4b656d31e1 Fix bug in profile avatar display
All checks were successful
ci/woodpecker/push/woodpecker Pipeline was successful
2023-04-28 16:10:02 +00:00
2f21d8fe2c Allow for un-setting uploaded avatars 2023-04-28 16:00:55 +00:00
196587616a Navbar avatar UI enhancements 2023-04-27 16:41:51 +00:00
22c80781d4 Added and basic usage of boring-avatars package 2023-04-27 16:32:28 +00:00
13 changed files with 97 additions and 48 deletions

View File

@ -3,7 +3,7 @@ pipeline:
group: build group: build
image: node image: node
when: when:
path: "web/*" path: "web/**/*"
environment: environment:
- VITE_API_URL=https://api.treadl.com - VITE_API_URL=https://api.treadl.com
- VITE_IMAGINARY_URL=https://images.treadl.com - VITE_IMAGINARY_URL=https://images.treadl.com
@ -22,7 +22,7 @@ pipeline:
image: woodpeckerci/plugin-docker-buildx image: woodpeckerci/plugin-docker-buildx
secrets: [docker_username, docker_password] secrets: [docker_username, docker_password]
when: when:
path: "api/*" path: "api/**/*"
settings: settings:
repo: wilw/treadl-api repo: wilw/treadl-api
dockerfile: api/Dockerfile dockerfile: api/Dockerfile
@ -32,7 +32,7 @@ pipeline:
image: alpine image: alpine
secrets: [ LINODE_ACCESS_KEY, LINODE_SECRET_ACCESS_KEY, BUNNY_KEY ] secrets: [ LINODE_ACCESS_KEY, LINODE_SECRET_ACCESS_KEY, BUNNY_KEY ]
when: when:
path: "web/*" path: "web/**/*"
commands: commands:
- cd web - cd web
- apk update - apk update

View File

@ -7,10 +7,10 @@ from api import uploads, groups
def get_users(user): def get_users(user):
db = database.get_db() db = database.get_db()
if 'root' not in user.get('roles', []): raise util.errors.Forbidden('Not allowed') if 'root' not in user.get('roles', []): raise util.errors.Forbidden('Not allowed')
users = list(db.users.find({}, {'username': 1, 'avatar': 1, 'email': 1, 'createdAt': 1, 'lastSeenAt': 1, 'roles': 1, 'groups': 1}).sort('lastSeenAt', -1)) users = list(db.users.find({}, {'username': 1, 'avatar': 1, 'email': 1, 'createdAt': 1, 'lastSeenAt': 1, 'roles': 1, 'groups': 1}).sort('lastSeenAt', -1).limit(200))
group_ids = [] group_ids = []
for u in users: group_ids += u.get('groups', []) for u in users: group_ids += u.get('groups', [])
groups = list(db.groups.find({'_id': {'$in': group_ids}})) groups = list(db.groups.find({'_id': {'$in': group_ids}}, {'name': 1}))
projects = list(db.projects.find({}, {'name': 1, 'path': 1, 'user': 1})) projects = list(db.projects.find({}, {'name': 1, 'path': 1, 'user': 1}))
for u in users: for u in users:
if 'avatar' in u: if 'avatar' in u:

View File

@ -47,12 +47,14 @@ def update(user, username, data):
if db.users.find({'username': data['username'].lower()}).count(): if db.users.find({'username': data['username'].lower()}).count():
raise util.errors.BadRequest('A user with this username already exists') raise util.errors.BadRequest('A user with this username already exists')
data['username'] = data['username'].lower() data['username'] = data['username'].lower()
if 'avatar' in data and len(data['avatar']) > 3: # Not a default avatar if data.get('avatar') and len(data['avatar']) > 3: # Not a default avatar
def handle_cb(h): def handle_cb(h):
db.users.update_one({'_id': user['_id']}, {'$set': {'avatarBlurHash': h}}) db.users.update_one({'_id': user['_id']}, {'$set': {'avatarBlurHash': h}})
uploads.blur_image('users/' + str(user['_id']) + '/' + data['avatar'], handle_cb) uploads.blur_image('users/' + str(user['_id']) + '/' + data['avatar'], handle_cb)
updater = util.build_updater(data, allowed_keys) updater = util.build_updater(data, allowed_keys)
if updater: if updater:
if 'avatar' in updater.get('$unset', {}): # Also unset blurhash if removing avatar
updater['$unset']['avatarBlurHash'] = ''
db.users.update({'username': username}, updater) db.users.update({'username': username}, updater)
return get(user, data.get('username', username)) return get(user, data.get('username', username))

Binary file not shown.

Binary file not shown.

View File

@ -8,6 +8,7 @@
"@sentry/tracing": "^6.19.6", "@sentry/tracing": "^6.19.6",
"@vitejs/plugin-react": "^2.2.0", "@vitejs/plugin-react": "^2.2.0",
"blurhash": "^1.1.3", "blurhash": "^1.1.3",
"boring-avatars": "^1.7.0",
"eventlistener": "^0.0.1", "eventlistener": "^0.0.1",
"moment": "^2.22.2", "moment": "^2.22.2",
"pell": "^1.0.4", "pell": "^1.0.4",

View File

@ -20,8 +20,8 @@ const StyledNavBar = styled.div`
margin-top:5px; margin-top:5px;
} }
.nav-links{ .nav-links{
vertical-align:middle; display: flex;
margin-top:8px; align-items: center;
.ui.button{ .ui.button{
margin-right: 8px; margin-right: 8px;
} }
@ -104,7 +104,7 @@ function NavBar() {
return ( return (
<StyledNavBar> <StyledNavBar>
<Container style={{display:'flex', justifyContent: 'space-between'}}> <Container style={{display:'flex', justifyContent: 'space-between', alignItems: 'center'}}>
<Link to="/"><img alt={`${utils.appName()} logo`} src={logoLight} className="logo" /></Link> <Link to="/"><img alt={`${utils.appName()} logo`} src={logoLight} className="logo" /></Link>
{isAuthenticated {isAuthenticated
? ( ? (
@ -185,17 +185,13 @@ function NavBar() {
<Button size='small' icon='help' basic inverted onClick={e => dispatch(actions.app.openHelpModal(true))}/> <Button size='small' icon='help' basic inverted onClick={e => dispatch(actions.app.openHelpModal(true))}/>
</span> </span>
<Dropdown direction="left" pointing="top right" icon={null} style={{marginLeft: 10}} <Dropdown direction="left" pointing="top right" icon={null} style={{marginLeft: 10, marginTop: 5}}
trigger={<UserChip user={user} withoutLink avatarOnly />} trigger={<UserChip user={user} withoutLink avatarOnly />}
> >
<Dropdown.Menu style={{ minWidth: '200px', paddingTop: 10 }}> <Dropdown.Menu style={{ minWidth: '200px', paddingTop: 10 }}>
{user && {user &&
<Dropdown.Header as={Link} to={`/${user.username}`}> <Dropdown.Header as={Link} to={`/${user.username}`}>
<div style={{ <UserChip user={user} />
display: 'inline-block', width: 30, height: 30, borderRadius: '50%', backgroundSize: 'cover', backgroundPosition: 'center center', backgroundImage: `url(${utils.avatarUrl(user)})`, verticalAlign: 'middle', marginRight: 8,
}}
/>
<span>{user.username}</span>
</Dropdown.Header> </Dropdown.Header>
} }
{user?.isGoldSupporter && <Dropdown.Header><SupporterBadge type='gold' /></Dropdown.Header>} {user?.isGoldSupporter && <Dropdown.Header><SupporterBadge type='gold' /></Dropdown.Header>}

View File

@ -1,12 +1,18 @@
import React from 'react'; import React from 'react';
import { Link } from 'react-router-dom'; import { Link } from 'react-router-dom';
import styled from 'styled-components'; import styled from 'styled-components';
import Avatar from 'boring-avatars';
import utils from '../../utils/utils.js'; import utils from '../../utils/utils.js';
import SupporterBadge from './SupporterBadge'; import SupporterBadge from './SupporterBadge';
const Avatar = styled.div` const StyledChip = styled(Link)`
display: inline-block; position: relative;
display: flex;
align-items: center;
`;
const CustomAvatar = styled.div`
display: block;
position: relative; position: relative;
width: ${props => props.compact ? 15 : 30}px; width: ${props => props.compact ? 15 : 30}px;
height: ${props => props.compact ? 15 : 30}px; height: ${props => props.compact ? 15 : 30}px;
@ -24,21 +30,33 @@ const Badge = styled.div`
`; `;
const Username = styled.span` const Username = styled.span`
font-size: ${props => props.compact ? 10 : 12}px; font-size: ${props => props.compact ? 10 : 12}px;
margin-left: 5px;
color: black;
`; `;
function UserChip({ user, compact, meta, withoutLink, avatarOnly, style }) { function UserChip({ user, compact, meta, withoutLink, avatarOnly, style }) {
if (!user) return null; if (!user) return null;
return ( return (
<Link to={withoutLink ? '#' : `/${user.username}`} style={style || {}}> <div style={{display: 'inline-block'}}>
<Avatar compact={compact} user={user} avatarOnly={avatarOnly}> <StyledChip to={withoutLink ? '#' : `/${user.username}`} style={style || {}}>
{user.isGoldSupporter && <Badge><SupporterBadge compact type='gold' /></Badge>} {user.avatar ?
{user.isSilverSupporter && !user.isGoldSupporter && <Badge><SupporterBadge compact type='silver' /></Badge>} <CustomAvatar compact={compact} user={user} avatarOnly={avatarOnly} />
</Avatar> :
{!avatarOnly && <> <Avatar
<Username compact={compact}>{user.username}</Username> size={compact ? 15 : 30}
{meta && <span style={{color:'rgb(180,180,180)', fontSize: 10, marginLeft: 10}}>{meta}</span>} name={user.username}
</>} variant="beam"
</Link> colors={["#B2A4FF", "#FFB4B4", "#FFDEB4", "#FDF7C3"]}
/>
}
{user.isGoldSupporter && <Badge><SupporterBadge compact type='gold' /></Badge>}
{user.isSilverSupporter && !user.isGoldSupporter && <Badge><SupporterBadge compact type='silver' /></Badge>}
{!avatarOnly && <>
<Username compact={compact}>{user.username}</Username>
{meta && <span style={{color:'rgb(180,180,180)', fontSize: 10, marginLeft: 10}}>{meta}</span>}
</>}
</StyledChip>
</div>
); );
} }

View File

@ -1,14 +1,24 @@
import React from 'react'; import React from 'react';
import { Divider, Container, Grid, Menu } from 'semantic-ui-react'; import { Divider, Container, Grid, Menu, Message } from 'semantic-ui-react';
import { Outlet, NavLink } from 'react-router-dom'; import { Outlet, NavLink, Link } from 'react-router-dom';
import { useSelector } from 'react-redux';
function Settings() { function Settings() {
const { user } = useSelector(state => {
const user = state.users.users.filter(u => state.auth.currentUserId === u._id)[0];
return { user };
});
if (!user) return;
return ( return (
<Container style={{ marginTop: '40px' }}> <Container style={{ marginTop: '40px' }}>
<h2>Manage your account</h2> <h2>Manage your account</h2>
<Divider hidden /> <Divider hidden />
<Grid stackable> <Grid stackable>
<Grid.Column computer={4}> <Grid.Column computer={4}>
<Message size='tiny'>You can change your profile information on your <Link to={`/${user.username}/edit`}>Profile page.</Link></Message>
<Menu fluid vertical tabular> <Menu fluid vertical tabular>
<Menu.Item as={NavLink} to="/settings/identity" name="identity" icon="user secret" /> <Menu.Item as={NavLink} to="/settings/identity" name="identity" icon="user secret" />
<Menu.Item as={NavLink} to='/settings/notifications' content='Notifications' icon='envelope' /> <Menu.Item as={NavLink} to='/settings/notifications' content='Notifications' icon='envelope' />

View File

@ -21,7 +21,13 @@ function EditProfile() {
}); });
const updatePicture = (avatar) => { const updatePicture = (avatar) => {
api.users.update(profileUser.username, { avatar }, u => dispatch(actions.users.receive(u))); api.users.update(profileUser.username, { avatar }, u => {
if (!avatar) { // Needed to ensure the avatar is immediately unset
u.avatar = null;
u.avatarUrl = null;
}
dispatch(actions.users.receive(u))
});
}; };
const updateProfile = () => { const updateProfile = () => {
@ -57,16 +63,14 @@ function EditProfile() {
trigger={<Button basic color="yellow" icon="image" content="Choose an image" />} trigger={<Button basic color="yellow" icon="image" content="Choose an image" />}
accept="image/*" onComplete={f => updatePicture(f.storedName)} accept="image/*" onComplete={f => updatePicture(f.storedName)}
/> />
<h4>Or choose one of ours:</h4> {profileUser.avatar &&
{utils.defaultAvatars().map(a => ( <Button basic color="gray" icon="times" content="Remove image" onClick={() => {
<img utils.confirm(
alt="Default avatar" key={a.key} src={a.url} 'Really remove your image?',
style={{ 'Your profile will be given a default avatar.'
width: 40, height: 40, margin: 4, cursor: 'pointer', ).then(() => updatePicture(null), () => {});
}} }}/>
onClick={e => updatePicture(a.key)} }
/>
))}
<Divider hidden /> <Divider hidden />
<Form> <Form>

View File

@ -3,6 +3,7 @@ import { Helmet } from 'react-helmet';
import { Loader, Icon, List, Container, Card, Grid, Message } from 'semantic-ui-react'; import { Loader, Icon, List, Container, Card, Grid, Message } from 'semantic-ui-react';
import { Link, Outlet, useParams } from 'react-router-dom'; import { Link, Outlet, useParams } from 'react-router-dom';
import { useDispatch, useSelector } from 'react-redux'; import { useDispatch, useSelector } from 'react-redux';
import Avatar from 'boring-avatars';
import moment from 'moment'; import moment from 'moment';
import utils from '../../../utils/utils'; import utils from '../../../utils/utils';
import actions from '../../../actions'; import actions from '../../../actions';
@ -57,12 +58,21 @@ function Profile() {
<Card raised color="yellow"> <Card raised color="yellow">
<Card.Content> <Card.Content>
<div style={{textAlign: 'center'}}> <div style={{textAlign: 'center'}}>
<BlurrableImage {profileUser.avatar ?
src={utils.cropUrl(utils.avatarUrl(profileUser), 200, 200)} <BlurrableImage
blurHash={profileUser.avatarBlurHash} src={utils.cropUrl(utils.avatarUrl(profileUser), 200, 200)}
width={200} height={200} blurHash={profileUser.avatarBlurHash}
style={{ borderRadius: '50%' }} width={200} height={200}
/> style={{ borderRadius: '50%' }}
/>
:
<Avatar
size={200}
name={profileUser.username}
variant="beam"
colors={["#B2A4FF", "#FFB4B4", "#FFDEB4", "#FDF7C3"]}
/>
}
</div> </div>
<Card.Header>{profileUser.username}</Card.Header> <Card.Header>{profileUser.username}</Card.Header>
<Card.Meta> <Card.Meta>

View File

@ -42,8 +42,8 @@ const utils = {
return window.location.protocol + '//' + window.location.host + path; return window.location.protocol + '//' + window.location.host + path;
}, },
avatarUrl(user) { avatarUrl(user) {
const avatar = (user && user.avatar) || '9'; const avatar = user?.avatar;
if (avatar.length < 3) { if (avatar?.length < 3) {
const a = utils.defaultAvatars().filter(a => a.key === avatar)[0]; const a = utils.defaultAvatars().filter(a => a.key === avatar)[0];
return a && a.url; return a && a.url;
} }

View File

@ -1071,6 +1071,13 @@ __metadata:
languageName: node languageName: node
linkType: hard linkType: hard
"boring-avatars@npm:^1.7.0":
version: 1.7.0
resolution: "boring-avatars@npm:1.7.0"
checksum: 5f39a7b3a011ba8651c6bec22c453b42c58c0fc347b3dce5cfea7d4de4208b1296f1fb4efc66ebb598f805124a07c5b8e3b06766f5211e1deb545bbf889230b0
languageName: node
linkType: hard
"brace-expansion@npm:^1.1.7": "brace-expansion@npm:^1.1.7":
version: 1.1.11 version: 1.1.11
resolution: "brace-expansion@npm:1.1.11" resolution: "brace-expansion@npm:1.1.11"
@ -3548,6 +3555,7 @@ __metadata:
"@sentry/tracing": ^6.19.6 "@sentry/tracing": ^6.19.6
"@vitejs/plugin-react": ^2.2.0 "@vitejs/plugin-react": ^2.2.0
blurhash: ^1.1.3 blurhash: ^1.1.3
boring-avatars: ^1.7.0
eventlistener: ^0.0.1 eventlistener: ^0.0.1
moment: ^2.22.2 moment: ^2.22.2
pell: ^1.0.4 pell: ^1.0.4