Skip to content

Commit

Permalink
feat: easter language toggle (#289)
Browse files Browse the repository at this point in the history
  • Loading branch information
NoamGaash authored Dec 14, 2023
1 parent bd902de commit a9caac9
Show file tree
Hide file tree
Showing 8 changed files with 289 additions and 248 deletions.
17 changes: 15 additions & 2 deletions README.md
Original file line number Diff line number Diff line change
@@ -1,15 +1,28 @@
# Open bus ranking app

## Welcome!
This is the official repository of the open bus (תחב"צ פתוחה) project - also known as "ShameBus".
This is the official repository of the open bus (תחב"צ פתוחה / דאטאבוס) project - also known as "ShameBus".
[link to the project](https://open-bus-map-search.hasadna.org.il/dashboard)
- While in the site, type "storybook" (in lowercase) to see a secret 😉

Please feel free to submit pull requests and contribute to the project (see the "contribution" section).

## View video (Hebrew language):
[![video (hebrew) about the project](https://img.youtube.com/vi/6H6jkJCVhgk/0.jpg)](https://www.youtube.com/watch?v=6H6jkJCVhgk)

# Easter eggs
We've hidden a couple of fun surprises in our web app, just for you. Discovering them is as easy as typing a few magic words on your keyboard.

## How to Find the Easter Eggs
1. Open our [web app](https://open-bus-map-search.hasadna.org.il/dashboard)
2. **Unleash the Magic Words:**
To reveal the hidden gems, use your keyboard to type the following commands:

- **Type "storybook":**
Watch the magic unfold as you type "storybook" on your keyboard. You might just stumble upon our Storybook, a treasure trove of UI components showcasing the beauty and functionality of our app.
- **Type "english":**
Feel like switching up the language? Type "english" and see the language toggle in action. Our app is multilingual, and you can experience it by triggering this secret command.


## deployments

[![Netlify Status](https://api.netlify.com/api/v1/badges/d3ef62c2-b5bb-48ac-8299-71e5bd22b211/deploy-status)](https://app.netlify.com/sites/open-bus/deploys)
Expand Down
10 changes: 7 additions & 3 deletions src/App.tsx
Original file line number Diff line number Diff line change
Expand Up @@ -18,10 +18,10 @@ import { heIL as heILmui } from '@mui/x-date-pickers/locales'
import { ThemeProvider, createTheme } from '@mui/material'
import { AdapterMoment } from '@mui/x-date-pickers/AdapterMoment'
import { LocalizationProvider } from '@mui/x-date-pickers'

import { usePages } from './routes'
import { EasterEgg } from './pages/EasterEgg/EasterEgg'
import { usePages } from './routes'
import MainLayout from './layout'
import { Envelope } from './pages/EasterEgg/Envelope'

const theme = createTheme(
{
Expand Down Expand Up @@ -105,7 +105,11 @@ const App = () => {
const RoutedApp = () => (
<Router>
<App />
<EasterEgg />
<EasterEgg code="storybook">
<a href="/storybook/index.html">
<Envelope />
</a>
</EasterEgg>
</Router>
)
export default RoutedApp
11 changes: 3 additions & 8 deletions src/layout/sidebar/menu/Menu.tsx
Original file line number Diff line number Diff line change
Expand Up @@ -6,6 +6,7 @@ import { usePages } from 'src/routes'

import type { MenuProps } from 'antd'
import { Menu } from 'antd'
import { LanguageToggle } from 'src/pages/EasterEgg/LanguageToggle'

type MenuItem = Required<MenuProps>['items'][number]
function getItem(
Expand All @@ -23,18 +24,12 @@ function getItem(
}

const MainMenu = () => {
const { t, i18n } = useTranslation()
const { t } = useTranslation()
const pages = usePages()
const items: MenuItem[] = pages.map((itm) => {
return getItem(<Link to={t(itm.path)}>{t(itm.label)}</Link>, itm.path, itm.icon)
})
const [currentLanguage, setCurrentLanguage] = useState('en')

const handleChangeLanguage = () => {
const newLanguage = currentLanguage === 'en' ? 'he' : 'en'
setCurrentLanguage(newLanguage)
i18n.changeLanguage(newLanguage)
}
const location = useLocation()
const [current, setCurrent] = useState(
location.pathname === '/' || location.pathname === '' ? '/dashboard' : location.pathname,
Expand All @@ -60,7 +55,7 @@ const MainMenu = () => {
mode="inline"
items={items}
/>
{null && <button onClick={handleChangeLanguage}>Change Language</button>}
{<LanguageToggle />}
</>
)
}
Expand Down
1 change: 1 addition & 0 deletions src/locale/en.json
Original file line number Diff line number Diff line change
Expand Up @@ -70,6 +70,7 @@
"migdal_company": "\"A tower in the community\"",
"and_smaller_donors": "And other small contributions from my friends and fans of the workshop.",
"github_link": "Go to GitHub",
"Change Language":"שנה שפה",
"bug_title": "Title/Summary",
"bug_title_message": "Please enter a title/summary!",
"bug_description": "Description",
Expand Down
1 change: 1 addition & 0 deletions src/locale/he.json
Original file line number Diff line number Diff line change
Expand Up @@ -70,6 +70,7 @@
"and_smaller_donors": "ותרומות קטנות נוספות של ידידי ואוהדי הסדנא.",
"gaps_patterns_page_title": "דפוסי נסיעות שלא יצאו",
"github_link": "למעבר אל GitHub",
"Change Language":"Change Language",
"bug_title": "כותרת/סיכום",
"bug_title_message": "אנא הזן כותרת/סיכום!",
"bug_description": "תיאור",
Expand Down
240 changes: 5 additions & 235 deletions src/pages/EasterEgg/EasterEgg.tsx
Original file line number Diff line number Diff line change
@@ -1,233 +1,9 @@
import { useState } from 'react'
import styled from 'styled-components'
import { createContext, useState } from 'react'
import useKonami from 'use-konami'

const colors = {
primaryColor300: '#7ec1ff', // the prev color '#e95f55',
primaryColor400: '#1890ff', // the prev color '#e15349' ,
primaryColor500: '#317fc8', // the prev color '#cb5a5e',
primaryColor600: '#136fc5', // the prev color '#cf4a43',
}
const EnvelopeWrapper = styled.div`
.letter-image {
position: absolute;
top: 50%;
left: 50%;
width: 200px;
height: 200px;
-webkit-transform: translate(-50%, -50%);
-moz-transform: translate(-50%, -50%);
transform: translate(-50%, -50%);
cursor: pointer;
direction: ltr;
z-index: 999;
opacity: 1;
animation: fadeIn 1s ease-in-out;
&.fade-out {
opacity: 0;
animation: fadeOut 1s ease-in-out;
}
}
@keyframes fadeIn {
0% {
opacity: 0;
}
100% {
opacity: 1;
}
}
@keyframes fadeOut {
0% {
opacity: 1;
}
100% {
opacity: 0;
}
}
.animated-mail {
position: absolute;
height: 150px;
width: 200px;
-webkit-transition: 0.4s;
-moz-transition: 0.4s;
transition: 0.4s;
.body {
position: absolute;
bottom: 0;
width: 0;
height: 0;
border-style: solid;
border-width: 0 0 100px 200px;
border-color: transparent transparent ${colors.primaryColor300} transparent;
z-index: 2;
}
.top-fold {
position: absolute;
top: 50px;
width: 0;
height: 0;
border-style: solid;
border-width: 50px 100px 0 100px;
-webkit-transform-origin: 50% 0%;
-webkit-transition: transform 0.4s 0.4s, z-index 0.2s 0.4s;
-moz-transform-origin: 50% 0%;
-moz-transition: transform 0.4s 0.4s, z-index 0.2s 0.4s;
transform-origin: 50% 0%;
transition: transform 0.4s 0.4s, z-index 0.2s 0.4s;
border-color: ${colors.primaryColor600} transparent transparent transparent;
z-index: 2;
}
.back-fold {
position: absolute;
bottom: 0;
width: 200px;
height: 100px;
background: ${colors.primaryColor600};
z-index: 0;
}
.left-fold {
position: absolute;
bottom: 0;
width: 0;
height: 0;
border-style: solid;
border-width: 50px 0 50px 100px;
border-color: transparent transparent transparent ${colors.primaryColor400};
z-index: 2;
}
.letter {
left: 20px;
bottom: 0px;
position: absolute;
width: 160px;
height: 60px;
background: white;
z-index: 1;
overflow: hidden;
-webkit-transition: 0.4s 0.2s;
-moz-transition: 0.4s 0.2s;
transition: 0.4s 0.2s;
.letter-border {
height: 10px;
width: 100%;
background: repeating-linear-gradient(
-45deg,
${colors.primaryColor500},
${colors.primaryColor500} 8px,
transparent 8px,
transparent 18px
);
}
.letter-title {
margin-top: 10px;
margin-left: 5px;
height: 10px;
width: 40%;
background: ${colors.primaryColor500};
}
.letter-context {
margin-top: 10px;
margin-left: 5px;
height: 10px;
width: 20%;
background: ${colors.primaryColor500};
}
.letter-favicon {
display: flex;
justify-content: center;
}
.letter-stamp {
margin-top: 30px;
margin-left: 120px;
border-radius: 100%;
height: 30px;
width: 30px;
background: ${colors.primaryColor500};
opacity: 0.3;
}
}
}
.shadow {
position: absolute;
top: 200px;
left: 50%;
width: 400px;
height: 30px;
transition: 0.4s;
transform: translateX(-50%);
-webkit-transition: 0.4s;
-webkit-transform: translateX(-50%);
-moz-transition: 0.4s;
-moz-transform: translateX(-50%);
border-radius: 100%;
background: radial-gradient(rgba(0, 0, 0, 0.5), rgba(0, 0, 0, 0), rgba(0, 0, 0, 0));
}
.letter-image:hover {
.animated-mail {
transform: translateY(50px);
-webkit-transform: translateY(50px);
-moz-transform: translateY(50px);
}
.animated-mail .top-fold {
transition: transform 0.4s, z-index 0.2s;
transform: rotateX(180deg);
-webkit-transition: transform 0.4s, z-index 0.2s;
-webkit-transform: rotateX(180deg);
-moz-transition: transform 0.4s, z-index 0.2s;
-moz-transform: rotateX(180deg);
z-index: 0;
}
.animated-mail .letter {
height: 180px;
}
.shadow {
width: 250px;
}
}
`
const Envelope = ({ fade }: { fade: boolean }) => (
<EnvelopeWrapper>
<div className={fade ? 'letter-image fade-out' : 'letter-image'}>
<div className="animated-mail">
<div className="back-fold"></div>
<div className="letter">
<div className="letter-border"></div>
<div className="letter-title"></div>
<div className="letter-context"></div>
<div className="letter-favicon">
<img src="/favicon.ico" alt="busFavicon" />
</div>
<div className="letter-stamp">
<div className="letter-stamp-inner"></div>
</div>
</div>
<div className="top-fold"></div>
<div className="body"></div>
<div className="left-fold"></div>
</div>
<div className="shadow"></div>
</div>
</EnvelopeWrapper>
)
export const FadeContext = createContext(false)

export function EasterEgg() {
export function EasterEgg({ children, code }: { children?: React.ReactNode; code: string }) {
const [show, setShow] = useState(false)
const [fade, setFade] = useState(false)
useKonami({
Expand All @@ -241,13 +17,7 @@ export function EasterEgg() {
setFade(false)
}, 10000)
},
sequence: 'storybook'.split(''),
sequence: code.split(''),
})
return (
show && (
<a href="/storybook/index.html">
<Envelope fade={fade} />
</a>
)
)
return show && <FadeContext.Provider value={fade}>{children}</FadeContext.Provider>
}
Loading

0 comments on commit a9caac9

Please sign in to comment.