mirror of https://github.com/mue/mue.git
feat: achievement improvements
- Added achievement notification - Show locked achievements - Fixed quote padding - Fixed date picker layout Co-authored-by: David Ralph <me@davidcralph.co.uk> Co-authored-by: Isaac <contact@eartharoid.me>
This commit is contained in:
parent
59357357bb
commit
86f64dfc98
|
@ -51,6 +51,8 @@ input {
|
||||||
/* date picker */
|
/* date picker */
|
||||||
&[type='date'] {
|
&[type='date'] {
|
||||||
width: 260px;
|
width: 260px;
|
||||||
|
display: flex;
|
||||||
|
flex-flow: column;
|
||||||
|
|
||||||
@include themed {
|
@include themed {
|
||||||
background: t($modal-sidebar);
|
background: t($modal-sidebar);
|
||||||
|
|
|
@ -33,6 +33,12 @@
|
||||||
}
|
}
|
||||||
|
|
||||||
.achievements {
|
.achievements {
|
||||||
|
display: flex;
|
||||||
|
flex-flow: column;
|
||||||
|
gap: 15px;
|
||||||
|
}
|
||||||
|
|
||||||
|
.achievementsGrid {
|
||||||
display: grid;
|
display: grid;
|
||||||
grid-template-columns: repeat(auto-fit, minmax(250px, 1fr));
|
grid-template-columns: repeat(auto-fit, minmax(250px, 1fr));
|
||||||
grid-gap: 10px;
|
grid-gap: 10px;
|
||||||
|
|
|
@ -10,7 +10,7 @@ import { Header, CustomActions } from 'components/Layout/Settings';
|
||||||
|
|
||||||
import { saveFile } from 'utils/saveFile';
|
import { saveFile } from 'utils/saveFile';
|
||||||
|
|
||||||
import { translations, achievements } from 'utils/achievements';
|
import { getLocalisedAchievementData, achievements, checkAchievements } from 'utils/achievements';
|
||||||
|
|
||||||
class Stats extends PureComponent {
|
class Stats extends PureComponent {
|
||||||
constructor() {
|
constructor() {
|
||||||
|
@ -21,29 +21,10 @@ class Stats extends PureComponent {
|
||||||
};
|
};
|
||||||
}
|
}
|
||||||
|
|
||||||
getAchievements() {
|
updateAchievements() {
|
||||||
const achievements = this.state.achievements;
|
const achieved = checkAchievements(this.state.stats);
|
||||||
achievements.forEach((achievement) => {
|
|
||||||
switch (achievement.condition.type) {
|
|
||||||
case 'tabsOpened':
|
|
||||||
if (this.state.stats['tabs-opened'] >= achievement.condition.amount) {
|
|
||||||
achievement.achieved = true;
|
|
||||||
}
|
|
||||||
break;
|
|
||||||
case 'addonInstall':
|
|
||||||
if (this.state.stats.marketplace) {
|
|
||||||
if (this.state.stats.marketplace['install'] >= achievement.condition.amount) {
|
|
||||||
achievement.achieved = true;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
break;
|
|
||||||
default:
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
});
|
|
||||||
|
|
||||||
this.setState({
|
this.setState({
|
||||||
achievements,
|
achievements: achieved,
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -59,11 +40,13 @@ class Stats extends PureComponent {
|
||||||
|
|
||||||
resetStats() {
|
resetStats() {
|
||||||
localStorage.setItem('statsData', JSON.stringify({}));
|
localStorage.setItem('statsData', JSON.stringify({}));
|
||||||
|
localStorage.setItem('achievements', JSON.stringify(achievements));
|
||||||
this.setState({
|
this.setState({
|
||||||
stats: {},
|
stats: {},
|
||||||
|
achievements,
|
||||||
});
|
});
|
||||||
toast(variables.getMessage('toasts.stats_reset'));
|
toast(variables.getMessage('toasts.stats_reset'));
|
||||||
this.getAchievements();
|
this.updateAchievements();
|
||||||
this.forceUpdate();
|
this.forceUpdate();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -75,31 +58,21 @@ class Stats extends PureComponent {
|
||||||
saveFile(JSON.stringify(this.state.stats, null, 2), filename);
|
saveFile(JSON.stringify(this.state.stats, null, 2), filename);
|
||||||
}
|
}
|
||||||
|
|
||||||
getLocalisedAchievementData(id) {
|
|
||||||
const localised = translations[variables.languagecode][id] ||
|
|
||||||
translations.en_GB[id] || { name: id, description: '' };
|
|
||||||
|
|
||||||
return {
|
|
||||||
name: localised.name,
|
|
||||||
description: localised.description,
|
|
||||||
};
|
|
||||||
}
|
|
||||||
|
|
||||||
componentDidMount() {
|
componentDidMount() {
|
||||||
this.getAchievements();
|
this.updateAchievements();
|
||||||
this.forceUpdate();
|
this.forceUpdate();
|
||||||
}
|
}
|
||||||
|
|
||||||
render() {
|
render() {
|
||||||
const achievementElement = (key, id, achieved) => {
|
const achievementElement = (key, id, achieved) => {
|
||||||
const { name, description } = this.getLocalisedAchievementData(id);
|
const { name, description } = getLocalisedAchievementData(id);
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<div className="achievement" key={key}>
|
<div className="achievement" key={key}>
|
||||||
<FaTrophy />
|
<FaTrophy />
|
||||||
<div className={'achievementContent' + (achieved ? ' achieved' : '')}>
|
<div className={'achievementContent' + (achieved ? ' achieved' : '')}>
|
||||||
<span>{name}</span>
|
<span>{name}</span>
|
||||||
<span className="subtitle">{description}</span>
|
<span className="subtitle">{achieved ? description : '?????'}</span>
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
);
|
);
|
||||||
|
@ -213,11 +186,21 @@ class Stats extends PureComponent {
|
||||||
</span>
|
</span>
|
||||||
</div>
|
</div>
|
||||||
<div className="achievements">
|
<div className="achievements">
|
||||||
{this.state.achievements.map((achievement, index) => {
|
<div className="achievementsGrid">
|
||||||
if (achievement.achieved) {
|
{this.state.achievements.map((achievement, index) => {
|
||||||
return achievementElement(index, achievement.id, achievement.achieved);
|
if (achievement.achieved) {
|
||||||
}
|
return achievementElement(index, achievement.id, achievement.achieved);
|
||||||
})}
|
}
|
||||||
|
})}
|
||||||
|
</div>
|
||||||
|
<span className="title">Locked</span>
|
||||||
|
<div className="achievementsGrid preferencesInactive">
|
||||||
|
{this.state.achievements.map((achievement, index) => {
|
||||||
|
if (!achievement.achieved) {
|
||||||
|
return achievementElement(index, achievement.id, achievement.achieved);
|
||||||
|
}
|
||||||
|
})}
|
||||||
|
</div>
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
</>
|
</>
|
||||||
|
|
|
@ -493,22 +493,27 @@ class Quote extends PureComponent {
|
||||||
<span className="subtitle">loading</span>
|
<span className="subtitle">loading</span>
|
||||||
</div>
|
</div>
|
||||||
)}
|
)}
|
||||||
<div className="quote-buttons">
|
{(this.state.authorOccupation !== 'Unknown' && this.state.authorlink !== null) ||
|
||||||
{this.state.authorOccupation !== 'Unknown' && this.state.authorlink !== null ? (
|
this.state.copy ||
|
||||||
<Tooltip title={variables.getMessage('widgets.quote.link_tooltip')}>
|
this.state.share ||
|
||||||
<a
|
this.state.favourited ? (
|
||||||
href={this.state.authorlink}
|
<div className="quote-buttons">
|
||||||
className="quoteAuthorLink"
|
{this.state.authorOccupation !== 'Unknown' && this.state.authorlink !== null ? (
|
||||||
target="_blank"
|
<Tooltip title={variables.getMessage('widgets.quote.link_tooltip')}>
|
||||||
rel="noopener noreferrer"
|
<a
|
||||||
aria-label="Learn about the author of the quote."
|
href={this.state.authorlink}
|
||||||
>
|
className="quoteAuthorLink"
|
||||||
<MdOpenInNew />
|
target="_blank"
|
||||||
</a>{' '}
|
rel="noopener noreferrer"
|
||||||
</Tooltip>
|
aria-label="Learn about the author of the quote."
|
||||||
) : null}
|
>
|
||||||
{this.state.copy} {this.state.share} {this.state.favourited}
|
<MdOpenInNew />
|
||||||
</div>
|
</a>{' '}
|
||||||
|
</Tooltip>
|
||||||
|
) : null}
|
||||||
|
{this.state.copy} {this.state.share} {this.state.favourited}
|
||||||
|
</div>
|
||||||
|
) : null}
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
)}
|
)}
|
||||||
|
|
|
@ -47,4 +47,4 @@
|
||||||
"name": "System Overload",
|
"name": "System Overload",
|
||||||
"description": "Installed 50 add-ons"
|
"description": "Installed 50 add-ons"
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -0,0 +1,50 @@
|
||||||
|
import { achievements } from './index';
|
||||||
|
|
||||||
|
export function checkAchievements(stats) {
|
||||||
|
achievements.forEach((achievement) => {
|
||||||
|
switch (achievement.condition.type) {
|
||||||
|
case 'tabsOpened':
|
||||||
|
if (stats['tabs-opened'] >= achievement.condition.amount) {
|
||||||
|
achievement.achieved = true;
|
||||||
|
}
|
||||||
|
break;
|
||||||
|
case 'addonInstall':
|
||||||
|
if (stats.marketplace) {
|
||||||
|
if (stats.marketplace['install'] >= achievement.condition.amount) {
|
||||||
|
achievement.achieved = true;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
break;
|
||||||
|
default:
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
return achievements;
|
||||||
|
}
|
||||||
|
|
||||||
|
export function newAchievements(stats) {
|
||||||
|
// calculate new achievements
|
||||||
|
const oldAchievements = JSON.parse(localStorage.getItem('achievements')) || [];
|
||||||
|
const checkedAchievements = checkAchievements(stats);
|
||||||
|
const newAchievements = [];
|
||||||
|
|
||||||
|
checkedAchievements.forEach((achievement) => {
|
||||||
|
if (achievement.achieved && !oldAchievements.includes(achievement.id)) {
|
||||||
|
newAchievements.push(achievement);
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
// add timestamp to new achievements
|
||||||
|
newAchievements.forEach((achievement) => {
|
||||||
|
achievement.timestamp = Date.now();
|
||||||
|
});
|
||||||
|
|
||||||
|
// save new achievements to local storage
|
||||||
|
localStorage.setItem(
|
||||||
|
'achievements',
|
||||||
|
JSON.stringify([...oldAchievements, ...newAchievements.map((achievement) => achievement.id)]),
|
||||||
|
);
|
||||||
|
|
||||||
|
return newAchievements;
|
||||||
|
}
|
|
@ -1,3 +1,5 @@
|
||||||
|
import variables from 'config/variables';
|
||||||
|
|
||||||
import de_DE from 'i18n/locales/achievements/de_DE.json';
|
import de_DE from 'i18n/locales/achievements/de_DE.json';
|
||||||
import en_GB from 'i18n/locales/achievements/en_GB.json';
|
import en_GB from 'i18n/locales/achievements/en_GB.json';
|
||||||
import en_US from 'i18n/locales/achievements/en_US.json';
|
import en_US from 'i18n/locales/achievements/en_US.json';
|
||||||
|
@ -14,6 +16,8 @@ import pt_BR from 'i18n/locales/achievements/pt_BR.json';
|
||||||
|
|
||||||
import achievements from 'utils/data/achievements.json';
|
import achievements from 'utils/data/achievements.json';
|
||||||
|
|
||||||
|
import { checkAchievements, newAchievements } from './condition';
|
||||||
|
|
||||||
const translations = {
|
const translations = {
|
||||||
de_DE,
|
de_DE,
|
||||||
en_GB,
|
en_GB,
|
||||||
|
@ -30,7 +34,15 @@ const translations = {
|
||||||
pt_BR,
|
pt_BR,
|
||||||
};
|
};
|
||||||
|
|
||||||
export {
|
// todo: clean this up and condition.js too
|
||||||
achievements,
|
function getLocalisedAchievementData(id) {
|
||||||
translations
|
const localised = translations[variables.languagecode][id] ||
|
||||||
};
|
translations.en_GB[id] || { name: id, description: '' };
|
||||||
|
|
||||||
|
return {
|
||||||
|
name: localised.name,
|
||||||
|
description: localised.description,
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
export { achievements, checkAchievements, newAchievements, getLocalisedAchievementData };
|
||||||
|
|
|
@ -42,7 +42,7 @@
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
"id": "808sandtabs",
|
"id": "808s",
|
||||||
"condition": {
|
"condition": {
|
||||||
"type": "tabsOpened",
|
"type": "tabsOpened",
|
||||||
"amount": 808
|
"amount": 808
|
||||||
|
|
|
@ -1,4 +1,17 @@
|
||||||
|
import { newAchievements, getLocalisedAchievementData } from './achievements';
|
||||||
|
import { toast } from 'react-toastify';
|
||||||
|
|
||||||
export default class Stats {
|
export default class Stats {
|
||||||
|
static async achievementTrigger(stats) {
|
||||||
|
const newAchievement = newAchievements(stats);
|
||||||
|
newAchievement.forEach((achievement) => {
|
||||||
|
if (achievement) {
|
||||||
|
const { name } = getLocalisedAchievementData(achievement.id);
|
||||||
|
toast(`Achievement Unlocked: ${name}`);
|
||||||
|
}
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* It takes two arguments, a type and a name, and then it increments the value of the name in the type
|
* It takes two arguments, a type and a name, and then it increments the value of the name in the type
|
||||||
* object in localStorage
|
* object in localStorage
|
||||||
|
@ -24,6 +37,7 @@ export default class Stats {
|
||||||
data[type][value] = data[type][value] + 1;
|
data[type][value] = data[type][value] + 1;
|
||||||
}
|
}
|
||||||
localStorage.setItem('statsData', JSON.stringify(data));
|
localStorage.setItem('statsData', JSON.stringify(data));
|
||||||
|
this.achievementTrigger(data);
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
@ -33,5 +47,6 @@ export default class Stats {
|
||||||
const data = JSON.parse(localStorage.getItem('statsData'));
|
const data = JSON.parse(localStorage.getItem('statsData'));
|
||||||
data['tabs-opened'] = data['tabs-opened'] + 1 || 1;
|
data['tabs-opened'] = data['tabs-opened'] + 1 || 1;
|
||||||
localStorage.setItem('statsData', JSON.stringify(data));
|
localStorage.setItem('statsData', JSON.stringify(data));
|
||||||
|
this.achievementTrigger(data);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
Loading…
Reference in New Issue