Compare commits
4 Commits
57032a60f0
...
4b656d31e1
Author | SHA1 | Date | |
---|---|---|---|
4b656d31e1 | |||
2f21d8fe2c | |||
196587616a | |||
22c80781d4 |
@ -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
|
||||||
|
@ -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:
|
||||||
|
@ -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))
|
||||||
|
|
||||||
|
BIN
web/.yarn/cache/boring-avatars-npm-1.7.0-b7d89b1ead-5f39a7b3a0.zip
vendored
Normal file
BIN
web/.yarn/cache/boring-avatars-npm-1.7.0-b7d89b1ead-5f39a7b3a0.zip
vendored
Normal file
Binary file not shown.
Binary file not shown.
@ -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",
|
||||||
|
@ -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>}
|
||||||
|
@ -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>
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -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' />
|
||||||
|
@ -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>
|
||||||
|
@ -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>
|
||||||
|
@ -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;
|
||||||
}
|
}
|
||||||
|
@ -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
|
||||||
|
Loading…
Reference in New Issue
Block a user