feat: update site
This commit is contained in:
		
							parent
							
								
									71b2aaae4f
								
							
						
					
					
						commit
						a6ff285b2e
					
				
					 6 changed files with 321 additions and 37 deletions
				
			
		
							
								
								
									
										
											BIN
										
									
								
								public/assets/logo.webp
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										
											BIN
										
									
								
								public/assets/logo.webp
									
										
									
									
									
										Normal file
									
								
							
										
											Binary file not shown.
										
									
								
							| After Width: | Height: | Size: 178 KiB | 
|  | @ -3,67 +3,104 @@ | |||
| <head> | ||||
|     <meta charset="UTF-8"> | ||||
|     <meta name="viewport" content="width=device-width, initial-scale=1.0"> | ||||
|     <meta name="description" content="Nebulosus, a company focused on providing an intuitive interface for anything and anyone. Making applications that are useful."> | ||||
|     <meta name="keywords" content="website,software,developer,nederland,web"> | ||||
|     <title>Nebulosus</title> | ||||
| 
 | ||||
|     <meta name="description" content="Nebulosus is a dutch company focusing on intuitivism, simplicity, redundancy, reliability and friendliness."> | ||||
|     <link rel="icon" href="/assets/logo.webp" type="image/webp"> | ||||
|     <link rel="stylesheet" href="./res/style.css"> | ||||
|     <script src="./res/cases.js" defer></script> | ||||
|     <script src="./res/anchors.js" defer></script> | ||||
|     <script src="./res/locale.js" defer></script> | ||||
| </head> | ||||
| 
 | ||||
| <style>  | ||||
|     [data-i18n-show] { | ||||
|         display: none; | ||||
|     } | ||||
| </style> | ||||
| 
 | ||||
| <body> | ||||
|     <header style="display: flex; justify-content: space-between;"> | ||||
|         <div class="logo" style="display: flex; align-items: center; gap: .75rem"> | ||||
|         <div class="logo" style="display: flex; align-items: center; gap: .75rem;"> | ||||
|             <img src="/assets/logo.webp" height="40" width="40" alt="" /> | ||||
|             <span>Nebulosus</span> | ||||
|         </div> | ||||
|         <div class="columns" style="padding: 1rem; justify-content: center;"> | ||||
|         <div class="columns" style="padding: 1.5rem; justify-content: center;"> | ||||
|             <a href="/">Home</a> | ||||
|         </div> | ||||
|     </header> | ||||
|     <main> | ||||
|     <main data-i18n-show> | ||||
|         <section id="intro"> | ||||
|             <div class="columns" style="align-items: flex-start; padding: 2rem; gap: 2rem;"> | ||||
|                 <img src="/assets/logo.webp" height="100" width="100" style="filter: drop-shadow(0 0 20px #fff);" /> | ||||
|                 <div class="rows"> | ||||
|                     <h1>Nebulosus</h1> | ||||
|                     <span>Focused on innovation.</span> | ||||
|                     <span data-i18n="$.motto"></span> | ||||
|                 </div> | ||||
|             </div> | ||||
|         </section> | ||||
|         <div> | ||||
|             <section id="offer"> | ||||
|                 <h2>What does Nebulosus offer?</h2> | ||||
|                 <span>Nebulosus offers a variety of services:</span> | ||||
|                 <ol class="rows" style="gap: .5em;"> | ||||
|                     <li><span style="border-bottom: 1px solid blue">Website / Application Development</span></li> | ||||
|                     <li><span style="border-bottom: 1px solid darkcyan">Server Hosting</span></li> | ||||
|                     <li><span style="border-bottom: 1px solid darkblue">Technical Support</span></li> | ||||
|                 </ol> | ||||
|                 <button id="cases-button" hidden>Cases</button> | ||||
|                 <div id="cases" hidden></div> | ||||
|             </section> | ||||
|             <section id="about"> | ||||
|                 <h2>About Nebulosus</h2> | ||||
|                 <p> | ||||
|                     Nebulosus is a dutch company focusing on intuitivism, <span style="letter-spacing: 1px;">simplicity</span>, <b style="color: darkgoldenrod;">redundancy</b>, <span style="color: darkcyan; text-decoration: underline;">reliability</span> and <i style="color: green">friendliness</i>. <br /> | ||||
|                     Software should be intuitive, aware of it's environment. That's what Nebulosus strives for. | ||||
|                     <br /> | ||||
|                     If you want to develop an app with intuitive design and a redundant infrastructure, please reach out! <br/> | ||||
|                     If you're looking for a company site, let's discuss it! | ||||
|                 </p> | ||||
|             </section> | ||||
|         </div> | ||||
|         <section id="more-info"> | ||||
|             <h2>More information about Nebulosus</h2> | ||||
| 
 | ||||
|         <!-- keep some sample text for SEO n stuff --> | ||||
|         <section id="about"> | ||||
|             <h2 data-i18n="about.title">About</h2> | ||||
|             <p data-i18n="about.text"> | ||||
|                 Nebulosus is a dutch company focusing on intuitivism, simplicity, redundancy, reliability and | ||||
|                 friendliness. | ||||
|                 Software should be intuitive, aware of it's environment. That's what Nebulosus strives for. | ||||
|                 If you want to develop an app with intuitive design and a redundant infrastructure, please reach out! | ||||
|                 If you're looking for a company site, let's discuss it! | ||||
|             </p> | ||||
|         </section> | ||||
| 
 | ||||
|         <section id="offer"> | ||||
|             <h2 data-i18n="offer.title">Offer</h2> | ||||
|             <div style="display: grid; gap: 1rem; grid-template-columns: repeat(auto-fill, minmax(300px, 1fr));"> | ||||
|                 <div> | ||||
|                     <h4 data-i18n="offer.it-consultancy.title"></h4> | ||||
|                     <p data-i18n="offer.it-consultancy.tagline"></p> | ||||
|                     <p data-i18n="offer.it-consultancy.description"></p> | ||||
|                     <span data-i18n="offer.it-consultancy.solutions"></span> | ||||
|                 </div> | ||||
|                 <div> | ||||
|                     <h4 data-i18n="offer.managed-hosting.title"></h4> | ||||
|                     <p data-i18n="offer.managed-hosting.tagline"></p> | ||||
|                     <p data-i18n="offer.managed-hosting.description"></p> | ||||
|                     <span data-i18n="offer.managed-hosting.solutions"></span> | ||||
|                 </div> | ||||
|                 <div> | ||||
|                     <h4 data-i18n="offer.tech-support.title"></h4> | ||||
|                     <p data-i18n="offer.tech-support.tagline"></p> | ||||
|                     <p data-i18n="offer.tech-support.description"></p> | ||||
|                     <span data-i18n="offer.tech-support.solutions"></span> | ||||
|                 </div> | ||||
|                 <div> | ||||
|                     <h4 data-i18n="offer.development.title"></h4> | ||||
|                     <p data-i18n="offer.development.tagline"></p> | ||||
|                     <p data-i18n="offer.development.description"></p> | ||||
|                     <span data-i18n="offer.development.solutions"></span> | ||||
|                 </div> | ||||
|             </div> | ||||
|         </section> | ||||
| 
 | ||||
|         <section id="info"> | ||||
|             <h2>Information</h2> | ||||
|             <div class="rows"> | ||||
|                 <span>E-mail: <a href="mailto:info@nebulosus.nl">info@nebulosus.nl</a></span> | ||||
|                 <span>Chamber of Commerce number: <strong>93409583</strong></span> | ||||
|                 <span><span data-i18n="dict.mail"></span>: <a href="#" data-i18n="$.mail" | ||||
|                         onclick="window.open('mailto:info'+'@'+''+'nebulosus'+'.n'+'l', '_self')"></a></span> | ||||
|                 <span><span data-i18n="dict.kvk"></span>: <strong data-i18n="$.kvk" href="#"></strong></span> | ||||
|             </div> | ||||
|         </section> | ||||
|     </main> | ||||
| 
 | ||||
|     <footer> | ||||
|         <div style="display: flex; justify-content: space-between"> | ||||
|             <span>© 2025 Nebulosus</span> | ||||
|             <div> | ||||
|                 <button onclick="locale.setPreferredLocale('en')">EN</button> | ||||
|                 <button onclick="locale.setPreferredLocale('nl')">NL</button> | ||||
|             </div> | ||||
|         </div> | ||||
|     </footer> | ||||
| 
 | ||||
|     <!-- Want to collaborate? --> | ||||
|     <!-- E-Mail: didier (at) nebulosus (dot) nl --> | ||||
| </body> | ||||
|  |  | |||
							
								
								
									
										103
									
								
								public/res/locale.js
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										103
									
								
								public/res/locale.js
									
										
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,103 @@ | |||
| (() => { | ||||
|     const getLocale = () => { | ||||
|         const cacheLocale = localStorage.getItem('preferredLocale') | ||||
|         if (cacheLocale) return cacheLocale | ||||
|         localStorage.setItem('preferredLocale', navigator.language); | ||||
|         return locale | ||||
|     } | ||||
| 
 | ||||
|     const setPreferredLocale = (locale) => { | ||||
|         localStorage.setItem('preferredLocale', locale); | ||||
|         setLocale() | ||||
|     } | ||||
| 
 | ||||
|     const getLocaleURL = (locale = getLocale()) => { | ||||
|         const lang = locale.split('-')[0]; | ||||
|         return `/res/locale/${lang}.json` | ||||
|     } | ||||
| 
 | ||||
|     const loadLocale = async (locale = getLocale()) => { | ||||
|         // check localStorage
 | ||||
|         const cacheDate = localStorage.getItem(`locale-${locale}`) | ||||
|         if (cacheDate) { | ||||
|             if (Date.now() - cacheDate < 1000 * 60 * 60 * 24) { // 1 day
 | ||||
|                 return JSON.parse(localStorage.getItem(`locale-${locale}-data`)) | ||||
|             } | ||||
|         } | ||||
|         try { | ||||
|             const response = await fetch(getLocaleURL(locale), { cache: 'no-store' }) | ||||
|             if (response.ok) { | ||||
|                 const data = await response.json() | ||||
|                 localStorage.setItem(`locale-${locale}`, Date.now()) | ||||
|                 localStorage.setItem(`locale-${locale}-data`, JSON.stringify(data)) | ||||
|                 return data | ||||
|             } | ||||
|         } catch (error) { | ||||
|             console.error(error) | ||||
|             if (cacheDate) { // if error then return cache even if expired
 | ||||
|                 return JSON.parse(localStorage.getItem(`locale-${locale}-data`)) | ||||
|             } else { | ||||
|                 if (locale !== 'en') return await loadLocale('en'); | ||||
|             } | ||||
|         } | ||||
|         return {} | ||||
|     } | ||||
| 
 | ||||
|     const setLocale = async (locale = getLocale()) => { | ||||
|         const data = await loadLocale(locale) | ||||
|         document.querySelectorAll('[data-i18n]').forEach(async (element) => { | ||||
|             try { | ||||
|                 // main.intro.title => {main: {intro: {title: 'value'}}}
 | ||||
|                 const keys = element.getAttribute('data-i18n').split('.'); | ||||
|                 console.log(keys) | ||||
|                 let value = data; | ||||
|                 keys.forEach((key) => value = value[key]); | ||||
|                 console.log(value) | ||||
|                 // if array then into ul list
 | ||||
|                 if (Array.isArray(value)) { | ||||
|                     element.innerHTML = '' | ||||
|                     const ul = document.createElement('ul') | ||||
|                     value.forEach((item) => { | ||||
|                         const li = document.createElement('li') | ||||
|                         li.innerHTML = item | ||||
|                         ul.appendChild(li) | ||||
|                     }) | ||||
|                     element.appendChild(ul) | ||||
|                 } else { | ||||
|                     element.innerHTML = value ?? '###'; | ||||
|                 } | ||||
|             } catch (error) { | ||||
|                 console.error(error) | ||||
|                 element.innerHTML = '###'; | ||||
|             } | ||||
|         }); | ||||
|         document.querySelectorAll('[data-i18n-show').forEach(el => el.removeAttribute('data-i18n-show')) | ||||
|     } | ||||
| 
 | ||||
|     window.locale = { | ||||
|         setPreferredLocale, | ||||
|         getLocale, | ||||
|         getLocaleURL, | ||||
|         loadLocale, | ||||
|         setLocale, | ||||
|         expireLocale: () => { | ||||
|             localStorage.removeItem(`locale-${getLocale()}`) | ||||
|             localStorage.removeItem(`locale-${getLocale()}-data`) | ||||
|             setLocale() | ||||
|         } | ||||
|     } | ||||
|     window.locale.setLocale(); | ||||
| 
 | ||||
|     // expiry management
 | ||||
|     // check if locale is older than EXPIRY_TIME then fetch again
 | ||||
|     function checkExpiry() { | ||||
|         const cacheDate = localStorage.getItem(`locale-${getLocale()}`) | ||||
|         if (cacheDate) { | ||||
|             if (Date.now() - cacheDate > 1000 * 60 * 60 * 24) { // 1 day
 | ||||
|                 window.locale.expireLocale() | ||||
|             } | ||||
|         } | ||||
|     } | ||||
|     setInterval(checkExpiry, 1000 * 60 * 60); // 1 hour
 | ||||
|     checkExpiry() | ||||
| })() | ||||
							
								
								
									
										66
									
								
								public/res/locale/en.json
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										66
									
								
								public/res/locale/en.json
									
										
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,66 @@ | |||
| { | ||||
|     "$": { | ||||
|         "name": "Nebulosus", | ||||
|         "kvk": 93409583, | ||||
|         "mail": "info@nebulosus.nl", | ||||
|         "motto": "Focused on innovation." | ||||
|     }, | ||||
|     "about": { | ||||
|         "title": "About", | ||||
|         "text": "Nebulosus is a Dutch company focused on intuition, <span style='letter-spacing: 1px;'>simplicity</span>, <b style='color: darkgoldenrod;'>redundancy</b>, <span style='color: darkcyan; text-decoration: underline;'>reliability</span>, and <i style='color: green'>friendliness</i>. <br /> We believe software should be intuitive and aware of its environment. This is the standard we aim for at Nebulosus.<br />Looking to develop an app with intuitive design and robust infrastructure? Get in touch!<br />Need a company website? Let’s discuss how we can help." | ||||
|     }, | ||||
|     "offer": { | ||||
|         "title": "Offer", | ||||
|         "it-consultancy": { | ||||
|             "title": "IT Consultancy", | ||||
|             "tagline": "IT consultancy services for your business.", | ||||
|             "description": "Nebulosus offers IT consultancy services for your business. Giving you a clear understanding of the technology you need to move forward.", | ||||
|             "solutions": [ | ||||
|                 "IT Strategy", | ||||
|                 "Infrastructure", | ||||
|                 "Security", | ||||
|                 "Compliance", | ||||
|                 "Training" | ||||
|             ] | ||||
|         }, | ||||
|         "managed-hosting": { | ||||
|             "title": "Managed Hosting", | ||||
|             "tagline": "Managed hosting services for your website.", | ||||
|             "description": "Nebulosus offers managed hosting services for your website. Taking care of the technical stuff so you can focus on your business.", | ||||
|             "solutions": [ | ||||
|                 "Application Hosting", | ||||
|                 "DNS Management (Domain and Subdomains)", | ||||
|                 "SSL Certificates", | ||||
|                 "Email Hosting" | ||||
|             ] | ||||
|         }, | ||||
|         "tech-support": { | ||||
|             "title": "Tech Support", | ||||
|             "tagline": "Professional support for your technical issues.", | ||||
|             "description": "Having an issue with your website, computer, or network? Nebulosus offers professional support for all your technical issues.", | ||||
|             "solutions": [ | ||||
|                 "Android/iOS Support", | ||||
|                 "Windows/Mac/Linux Support", | ||||
|                 "Network Support", | ||||
|                 "Installation and Configuration", | ||||
|                 "Advice and Guidance" | ||||
|             ] | ||||
|         }, | ||||
|         "development": { | ||||
|             "title": "Development", | ||||
|             "tagline": "Development services for your business.", | ||||
|             "description": "Nebulosus offers development services for your application. Whether you need a website, custom integration, or want to automate your business, Nebulosus can help.", | ||||
|             "solutions": [ | ||||
|                 "(Web)applications", | ||||
|                 "Integration development", | ||||
|                 "Automation", | ||||
|                 "API Development", | ||||
|                 "Database Development" | ||||
|             ] | ||||
|         } | ||||
|     }, | ||||
|     "dict": { | ||||
|         "kvk": "Chamber of Commerce", | ||||
|         "mail": "Email" | ||||
|     } | ||||
| } | ||||
							
								
								
									
										66
									
								
								public/res/locale/nl.json
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										66
									
								
								public/res/locale/nl.json
									
										
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,66 @@ | |||
| { | ||||
|     "$": { | ||||
|         "name": "Nebulosus", | ||||
|         "kvk": 93409583, | ||||
|         "mail": "info@nebulosus.nl", | ||||
|         "motto": "Gericht op innovatie." | ||||
|     }, | ||||
|     "about": { | ||||
|         "title": "Over", | ||||
|         "text": "Nebulosus is een Nederlands bedrijf dat streeft naar intuïtieve, eenvoudige en betrouwbare software, met redundantie en gebruiksvriendelijkheid als kernwaarden. <br /> Software moet intuïtief en omgevingsbewust zijn. Daar zetten wij ons voor in.<br />Wil je een app met intuïtief design en een robuuste infrastructuur? Neem contact op!<br />Op zoek naar een bedrijfssite? Laten we het bespreken!" | ||||
|     }, | ||||
|     "offer": { | ||||
|         "title": "Aanbod", | ||||
|         "it-consultancy": { | ||||
|             "title": "IT Consultancy", | ||||
|             "tagline": "IT-consultancydiensten voor uw bedrijf.", | ||||
|             "description": "Technologie moet je vooruithelpen, niet tegenwerken. Strategisch advies waarmee uw bedrijf wendbaar blijft in een snel veranderend digitaal landschap.", | ||||
|             "solutions": [ | ||||
|                 "IT-strategie", | ||||
|                 "Infrastructuur", | ||||
|                 "Beveiliging", | ||||
|                 "Compliance", | ||||
|                 "Training" | ||||
|             ] | ||||
|         }, | ||||
|         "managed-hosting": { | ||||
|             "title": "Beheerde Hosting", | ||||
|             "tagline": "Beheerde hostingdiensten voor uw website.", | ||||
|             "description": "Nebulosus biedt beheerde hostingdiensten voor uw website. Terwijl de technische zaken worden geregeld, kunt u zich richten op uw bedrijf.", | ||||
|             "solutions": [ | ||||
|                 "Applicatie Hosting", | ||||
|                 "DNS-beheer (Domeinnaam en subdomeinen)", | ||||
|                 "SSL-certificaten", | ||||
|                 "E-mailhosting" | ||||
|             ] | ||||
|         }, | ||||
|         "tech-support": { | ||||
|             "title": "Technische Ondersteuning", | ||||
|             "tagline": "Professionele ondersteuning bij uw technische problemen.", | ||||
|             "description": "Een probleem met uw website, computer of netwerk? Nebulosus biedt professionele ondersteuning voor al uw technische problemen.", | ||||
|             "solutions": [ | ||||
|                 "Android/iOS-ondersteuning", | ||||
|                 "Windows/Mac/Linux-ondersteuning", | ||||
|                 "Netwerkondersteuning", | ||||
|                 "Installatie en configuratie", | ||||
|                 "Advies en begeleiding" | ||||
|             ] | ||||
|         }, | ||||
|         "development": { | ||||
|             "title": "Ontwikkeling", | ||||
|             "tagline": "Ontwikkelingsdiensten voor uw bedrijf.", | ||||
|             "description": "Nebulosus biedt ontwikkelingsdiensten voor uw applicatie. Of u nu een website, mobiele app of aangepaste software nodig heeft, Nebulosus kan helpen.", | ||||
|             "solutions": [ | ||||
|                 "(Web)applicaties", | ||||
|                 "Integratie ontwikkeling", | ||||
|                 "Automatisering", | ||||
|                 "API's", | ||||
|                 "Databaseontwikkeling" | ||||
|             ] | ||||
|         } | ||||
|     }, | ||||
|     "dict": { | ||||
|         "kvk": "Kamer van Koophandel", | ||||
|         "mail": "E-mail" | ||||
|     } | ||||
| } | ||||
|  | @ -25,7 +25,6 @@ header { | |||
|     color: #000; | ||||
|     align-content: center; | ||||
|     z-index: 100; | ||||
|     box-shadow: 0 0 20px var(--theme-color-lowlight); | ||||
| } | ||||
| 
 | ||||
| header div.logo { | ||||
|  | @ -80,6 +79,20 @@ div section { | |||
| 
 | ||||
| /* --- */ | ||||
| 
 | ||||
| footer { | ||||
|     display: flex; | ||||
|     flex-direction: column; | ||||
|     position: fixed; | ||||
|     bottom: 0; | ||||
|     width: calc(100vw - 5em); | ||||
|     height: 2rem; | ||||
|     padding-left: 2.5em; | ||||
|     background-color: #fff; | ||||
|     justify-content: center; | ||||
| } | ||||
| 
 | ||||
| /* --- */ | ||||
| 
 | ||||
| /* for anchors.js */ | ||||
| h2:hover { | ||||
|     cursor: pointer; | ||||
|  | @ -148,7 +161,7 @@ button { | |||
|     cursor: pointer; | ||||
| 
 | ||||
|     &:hover { | ||||
|         background-color: var(--theme-color-highlight-2); | ||||
|         border: 1px solid var(--theme-color-highlight); | ||||
|     } | ||||
| 
 | ||||
|     transition: background-color .2s ease-in-out; | ||||
|  | @ -156,7 +169,6 @@ button { | |||
| 
 | ||||
| .columns { | ||||
|     display: flex; | ||||
|     align-items: center; | ||||
| 
 | ||||
|     @media (max-width: 600px) { | ||||
|         flex-direction: column; | ||||
|  |  | |||
		Loading…
	
	Add table
		Add a link
		
	
		Reference in a new issue