@networkpro/web 1.18.5 → 1.20.0

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (68) hide show
  1. package/.github/workflows/build-and-publish.yml +5 -4
  2. package/.github/workflows/lighthouse.yml +1 -1
  3. package/.github/workflows/meta-check.yml +1 -1
  4. package/.github/workflows/playwright.yml +1 -1
  5. package/.github/workflows/publish-test.yml +5 -4
  6. package/.github/workflows/templates/publish.template.yml +5 -4
  7. package/CHANGELOG.md +116 -1
  8. package/cspell.json +3 -0
  9. package/package.json +16 -16
  10. package/scripts/auditScripts.js +1 -1
  11. package/scripts/bundleCss.js +1 -1
  12. package/scripts/checkEnv.js +1 -1
  13. package/scripts/checkNode.js +1 -1
  14. package/scripts/checkVersions.js +1 -1
  15. package/scripts/generateTest.js +1 -1
  16. package/scripts/openReport.js +1 -1
  17. package/src/app.html +3 -6
  18. package/src/hooks.server.js +10 -7
  19. package/src/lib/README.md +160 -0
  20. package/src/lib/components/FullWidthSection.svelte +0 -9
  21. package/src/lib/components/ServiceSummaryTable.svelte +113 -0
  22. package/src/lib/components/foss/index.js +21 -0
  23. package/src/lib/components/index.js +31 -0
  24. package/src/lib/components/layout/HeaderDefault.svelte +3 -1
  25. package/src/lib/components/layout/HeaderHome.svelte +6 -5
  26. package/src/lib/components/layout/index.js +21 -0
  27. package/src/lib/images.js +1 -1
  28. package/src/lib/index.js +15 -5
  29. package/src/lib/meta.js +14 -9
  30. package/src/lib/pages/AboutContent.svelte +51 -15
  31. package/src/lib/pages/HomeContent.svelte +11 -1
  32. package/src/lib/pages/LicenseContent.svelte +3 -3
  33. package/src/lib/pages/PrivacyContent.svelte +31 -1
  34. package/src/lib/pages/ServicesContent.svelte +545 -0
  35. package/src/lib/pages/index.js +28 -0
  36. package/src/lib/stores/posthog.js +2 -1
  37. package/src/lib/styles/css/default.css +95 -0
  38. package/src/lib/styles/global.min.css +1 -1
  39. package/src/lib/styles/index.js +0 -1
  40. package/src/lib/types/README.md +50 -0
  41. package/src/lib/types/appConstants.js +1 -1
  42. package/src/lib/types/fossTypes.js +1 -1
  43. package/src/lib/utils/getUTMParams.js +1 -1
  44. package/src/lib/utils/purify.js +1 -1
  45. package/src/lib/utils/redirect.js +1 -1
  46. package/src/lib/utils/utm.js +2 -2
  47. package/src/routes/+layout.svelte +3 -11
  48. package/src/routes/+page.svelte +9 -7
  49. package/src/routes/consultation/+page.svelte +1 -1
  50. package/src/routes/links/+page.svelte +3 -2
  51. package/src/routes/services/+page.server.js +18 -0
  52. package/src/routes/services/+page.svelte +65 -0
  53. package/src/routes/status/+page.server.js +1 -1
  54. package/src/service-worker.js +2 -2
  55. package/static/bin/contact.vcf +1 -2
  56. package/static/disableSw.js +1 -1
  57. package/static/sitemap.xml +17 -5
  58. package/tests/e2e/app.spec.js +1 -1
  59. package/tests/e2e/mobile.spec.js +1 -1
  60. package/tests/e2e/shared/helpers.js +1 -1
  61. package/tests/meta/meta.test.js +1 -1
  62. package/tests/unit/client/lib/PWAInstallButton.test.js +80 -0
  63. package/tests/unit/client/lib/utils/redirect.test.js +1 -1
  64. package/tests/unit/server/internal/auditCoverage.test.js +1 -1
  65. package/tests/unit/server/lib/utils/purify.test.js +1 -1
  66. package/tests/unit/server/meta.test.js +1 -1
  67. package/vercel.json +12 -0
  68. package/vitest-setup-client.js +9 -0
@@ -316,6 +316,80 @@ footer .container {
316
316
  }
317
317
  }
318
318
 
319
+ #service-summary {
320
+ color: #e6e6e6;
321
+ margin-bottom: 2.5rem;
322
+ margin-top: 2rem;
323
+ }
324
+
325
+ .service-table {
326
+ width: 100%;
327
+ font-size: 0.95rem;
328
+ color: #e6e6e6;
329
+ background-color: #191919;
330
+ border-collapse: collapse;
331
+ }
332
+
333
+ .service-table th,
334
+ .service-table td {
335
+ padding: 0.75rem 1rem;
336
+ text-align: left;
337
+ border-bottom: 1px solid #333;
338
+ vertical-align: top;
339
+ }
340
+
341
+ .service-table th {
342
+ font-size: 0.85rem;
343
+ font-weight: 600;
344
+ color: #ffc627;
345
+ background-color: #222;
346
+ text-transform: uppercase;
347
+ }
348
+
349
+ .service-table a {
350
+ font-weight: 500;
351
+ color: #ffc627;
352
+ text-decoration: none;
353
+ }
354
+
355
+ .service-table a:hover {
356
+ text-decoration: underline;
357
+ }
358
+
359
+ /* Default row hover with gold glow */
360
+ .service-table tr:hover {
361
+ box-shadow: 0 0 10px 2px rgba(255, 198, 39, 0.25);
362
+ background-color: #222;
363
+ transition:
364
+ background-color 0.3s ease,
365
+ box-shadow 0.3s ease;
366
+ }
367
+
368
+ /* Selected (active) row — gold border, no glow unless hovered */
369
+ .service-table tr.selected {
370
+ background-color: #222;
371
+ transition:
372
+ background-color 0.3s ease,
373
+ border-left-color 0.3s ease;
374
+ border-left: 4px solid #ffc627;
375
+ }
376
+
377
+ /* Add subtle glow when hovering over the active row */
378
+ .service-table tr.selected:hover {
379
+ box-shadow: 0 0 12px 3px rgba(255, 198, 39, 0.35);
380
+ background-color: #252525;
381
+ }
382
+
383
+ @media (max-width: 768px) {
384
+ .service-table {
385
+ font-size: 0.9rem;
386
+ }
387
+ .service-table th,
388
+ .service-table td {
389
+ padding: 0.5rem;
390
+ }
391
+ }
392
+
319
393
  /* ==========================================================================
320
394
  Custom CSS
321
395
  ========================================================================== */
@@ -469,6 +543,13 @@ footer .container {
469
543
  word-break: normal; /* avoid deprecated behavior */
470
544
  }
471
545
 
546
+ .full-width-section.centered {
547
+ display: flex;
548
+ justify-content: center;
549
+ flex-direction: column;
550
+ min-height: 80vh;
551
+ }
552
+
472
553
  .fingerprint {
473
554
  display: block;
474
555
  font-weight: bold;
@@ -614,3 +695,17 @@ footer .container {
614
695
  margin-left: 3px;
615
696
  vertical-align: text-bottom;
616
697
  }
698
+
699
+ #toc ul {
700
+ list-style-type: disc;
701
+ padding-left: 1.5rem;
702
+ }
703
+
704
+ #toc a {
705
+ color: var(--color-primary, #ffc627);
706
+ text-decoration: none;
707
+ }
708
+
709
+ #toc a:hover {
710
+ text-decoration: underline;
711
+ }
@@ -3,4 +3,4 @@ Copyright © 2025 Network Pro Strategies (Network Pro™)
3
3
  SPDX-License-Identifier: CC-BY-4.0 OR GPL-3.0-or-later
4
4
  This file is part of Network Pro.
5
5
  ========================================================================== */
6
- html{-webkit-text-size-adjust:100%}body{margin:0}main{display:block}h1{margin:.67em 0;font-size:2em}hr{box-sizing:content-box}pre{font-family:monospace;font-size:1em}a{background-color:#0000}abbr[title]{border-bottom:none;text-decoration:underline dotted}b,strong{font-weight:bolder}code,kbd,samp{font-family:monospace;font-size:1em}small{font-size:80%}sub,sup{vertical-align:baseline;font-size:75%;line-height:0;position:relative}sub{bottom:-.25em}sup{top:-.5em}img{border-style:none}button,input,optgroup,select,textarea{margin:0;font-family:inherit;font-size:100%;line-height:1.15}button,input{overflow:visible}button,select{text-transform:none}button,[type=button],[type=reset],[type=submit]{-webkit-appearance:button;appearance:button}button::-moz-focus-inner,[type=button]::-moz-focus-inner,[type=reset]::-moz-focus-inner,[type=submit]::-moz-focus-inner{border-style:none;padding:0}button:-moz-focusring,[type=button]:-moz-focusring,[type=reset]:-moz-focusring,[type=submit]:-moz-focusring{outline:1px dotted buttontext}fieldset{padding:.35em .75em .625em}legend{color:inherit;box-sizing:border-box;white-space:normal;max-width:100%;padding:0;display:table}progress{vertical-align:baseline}textarea{overflow:auto}[type=checkbox],[type=radio]{box-sizing:border-box;padding:0}[type=number]::-webkit-inner-spin-button,[type=number]::-webkit-outer-spin-button{height:auto}[type=search]{-webkit-appearance:textfield;appearance:textfield;outline-offset:-2px}[type=search]::-webkit-search-decoration{-webkit-appearance:none}::-webkit-file-upload-button{font:inherit;-webkit-appearance:button}details{display:block}summary{display:list-item}template{display:none}html{color:#222;scroll-behavior:smooth;font-size:1em;line-height:1.4}::-moz-selection{text-shadow:none;background:#191919}::selection{text-shadow:none;background:#191919}hr{border:0;border-top:1px solid #ccc;height:1px;margin:1em 0;padding:0;display:block;overflow:visible}audio,canvas,iframe,img,svg,video{vertical-align:middle}fieldset{border:0;margin:0;padding:0}textarea{resize:vertical}body{color:#fafafa;background-color:#191919;margin:10px;font-family:Arial,Helvetica,sans-serif}a{text-decoration:none}a:link{color:#ffc627}a:hover,a:active{color:#ffc627;text-decoration:underline}a:focus{color:#111;background-color:#ffc627}a:visited,a:visited:hover{color:#cba557}a:visited:focus,a:visited:focus-visible{color:#111!important}.hidden,[hidden]{display:none!important}.visually-hidden{clip:rect(0,0,0,0);white-space:nowrap;border:0;width:1px;height:1px;margin:-1px;padding:0;position:absolute;overflow:hidden}.visually-hidden.focusable:active,.visually-hidden.focusable:focus{clip:auto;width:auto;height:auto;white-space:inherit;margin:0;position:static;overflow:visible}.invisible{visibility:hidden}.clearfix:before,.clearfix:after{content:"";display:table}.clearfix:after{clear:both}@media print{*,:before,:after{color:#000!important;box-shadow:none!important;text-shadow:none!important;background:#fff!important}a,a:visited{text-decoration:underline}a[href]:after{content:" (" attr(href)")"}abbr[title]:after{content:" (" attr(title)")"}a[href^=\#]:after,a[href^=javascript\:]:after{content:""}pre{white-space:pre-wrap!important}pre,blockquote{break-inside:avoid;border:1px solid #999}tr,img{break-inside:avoid}p,h2,h3{orphans:3;widows:3}h2,h3{break-after:avoid}}.full-width-section{background-position:50%;background-size:cover;width:100%;max-width:1920px;margin:0 auto}.container{max-width:1200px;margin:0 auto;padding:0 12px}.readable{max-width:900px;margin:0 auto}header,footer{width:100%}header .container,footer .container{max-width:1200px;margin:0 auto;padding:20px 12px}.gh{border-collapse:collapse;border-spacing:0;margin:0 auto}.gh td,.gh th{border-collapse:collapse;word-break:normal;padding:10px 5px;overflow:hidden}.gh .gh-tcell{text-align:center;vertical-align:middle}@media screen and (width<=767px){.gh,.gh col{width:auto!important}.gh-wrap{-webkit-overflow-scrolling:touch;margin:auto 0;overflow-x:auto}}.soc{border-collapse:collapse;border-spacing:0;margin:0 auto}.soc td,.soc th{border-collapse:collapse;word-break:normal;padding:8px;overflow:hidden}.soc .soc-fa{text-align:center;vertical-align:middle}@media screen and (width<=767px){.soc,.soc col{width:auto!important}.soc-wrap{-webkit-overflow-scrolling:touch;margin:auto 0;overflow-x:auto}}.foss{border-collapse:collapse;border-spacing:0}.foss td,.foss th{border-collapse:collapse;word-break:normal;padding:10px 5px;overflow:hidden}.foss .foss-cell{text-align:center;vertical-align:middle}@media screen and (width<=767px){.foss,.foss col{width:auto!important}.foss-wrap{-webkit-overflow-scrolling:touch;overflow-x:auto}}.bnav{text-align:center;border-collapse:collapse;border-spacing:0;margin:0 auto}.bnav td,.bnav th{text-align:center;vertical-align:middle;word-break:normal;border-style:none;padding:10px;font-size:.875rem;font-weight:700;line-height:1.125rem;overflow:hidden}.bnav .bnav-cell{text-align:center;vertical-align:middle;align-content:center}@media screen and (width<=767px){.bnav,.bnav col{width:auto!important}.bnav-wrap{-webkit-overflow-scrolling:touch;margin:auto 0;overflow-x:auto}}.bnav2{border-collapse:collapse;border-spacing:0;margin:0 auto}.bnav2 td{word-break:normal;border-style:none;padding:10px;font-size:.875rem;font-weight:700;line-height:1.125rem;overflow:hidden}.bnav2 th{word-break:normal;border-style:none;padding:12px;font-size:.875rem;line-height:1.125rem;overflow:hidden}.bnav2 .bnav2-cell{text-align:center;vertical-align:middle;align-content:center}@media screen and (width<=767px){.bnav2,.bnav2 col{width:auto!important}.bnav2-wrap{-webkit-overflow-scrolling:touch;margin:auto 0;overflow-x:auto}}.pgp{border-collapse:collapse;border-spacing:0;margin:0 auto}.pgp td{word-break:normal;border-style:none;padding:10px;font-size:.875rem;line-height:1.125rem;overflow:hidden}.pgp th{word-break:normal;border:1px solid #000;padding:10px;font-size:.875rem;line-height:1.125rem;overflow:hidden}.pgp .pgp-col1{text-align:right;vertical-align:middle;padding-right:1rem}.pgp .pgp-col2{text-align:left;vertical-align:middle;padding-left:1rem}@media screen and (width<=767px){.pgp,.pgp col{width:auto!important}.pgp-wrap{-webkit-overflow-scrolling:touch;margin:2rem 0 auto;overflow-x:auto}}.logo{margin-left:auto;margin-right:auto;display:block}.index-title1{text-align:center;font-style:italic;font-weight:700}.index-title2{letter-spacing:-.015em;text-align:center;font-variant:small-caps;font-size:1.25rem;line-height:1.625rem}.index1{letter-spacing:-.035em;text-align:center;font-style:italic;font-weight:700;line-height:2.125rem}.index2{letter-spacing:-.035em;text-align:center;font-variant:small-caps;font-size:1.5rem;line-height:1.75rem}.index3{letter-spacing:-.035em;text-align:center;font-size:1.5rem;line-height:1.75rem}.index4{letter-spacing:-.035em;text-align:center;font-size:1.5rem;line-height:1.75rem;text-decoration:underline}.subhead{letter-spacing:-.035em;font-variant:small-caps;font-size:1.5rem;line-height:1.75rem}.bold{font-weight:700}.emphasis{font-style:italic}.uline{text-decoration:underline}.bolditalic{font-style:italic;font-weight:700}.bquote{border-left:3px solid #9e9e9e;margin-left:30px;padding-left:10px;font-style:italic}.small-text{font-size:.75rem;line-height:1.125rem}.large-text-center{text-align:center;font-size:1.25rem;line-height:1.75rem}.prewrap{white-space:pre-wrap;display:block}.hr-styled{width:75%;margin:auto}.center-text{text-align:center}.copyright{text-align:center;font-size:.75rem;line-height:1.125rem}.gold{color:#ffc627}.visited{color:#cba557}.goldseparator{color:#ffc627;margin:0 .5rem}.center-nav{text-align:center;padding:5px;font-size:1rem;line-height:1.5rem}.block{overflow-wrap:break-word;resize:none;white-space:normal;word-break:normal;background:0 0;border:none;border-radius:0;outline:none;width:100%;font-family:monospace;font-size:.875rem;line-height:1.125rem}.fingerprint{white-space:pre-line;font-weight:700;display:block}.pgp-image{width:150px;height:150px}.spacer{margin:2rem 0}.separator{margin:0 .5rem}.emoji{margin-right:8px}.headline{margin-bottom:4px;font-style:italic;font-weight:700;display:block}.label{font-family:inherit;font-weight:700}.description{font-family:inherit;font-style:normal;font-weight:400;display:inline}.sr-only{clip:rect(0,0,0,0);white-space:nowrap;border:0;width:1px;height:1px;margin:-1px;padding:0;position:absolute;overflow:hidden}.pgp-entry{flex-wrap:wrap;align-items:center;gap:2rem;margin-bottom:2rem;display:flex}.pgp-text{flex:2;min-width:250px}.pgp-qr{flex:1;min-width:150px}.obtainium-direct-label{margin:.25rem 0 .75rem;font-weight:700}.obtainium-manual-label{margin-top:.75rem;font-weight:700}.obtainium-img{width:185px;height:55px;margin-bottom:.25rem}.obtainium-margin{margin-left:4px}.obtainium-fa-down{color:#ffc627;margin-left:4px}.obtainium-icon{width:50px;height:50px}.proton-img{width:168px;height:24px}.redirect-text{text-align:center;margin-top:4rem}.redirect-content{text-align:center;margin-top:2rem;font-family:system-ui,sans-serif}.loading-spinner{border:4px solid #ddd;border-top-color:#ffc627;border-radius:50%;width:48px;height:48px;margin:2rem auto;animation:1s linear infinite spin}@keyframes spin{to{transform:rotate(360deg)}}.cc-link{text-decoration:none}.cc-img{vertical-align:text-bottom;margin-left:3px;display:inline-block;height:18px!important}
6
+ html{-webkit-text-size-adjust:100%}body{margin:0}main{display:block}h1{margin:.67em 0;font-size:2em}hr{box-sizing:content-box}pre{font-family:monospace;font-size:1em}a{background-color:#0000}abbr[title]{border-bottom:none;text-decoration:underline dotted}b,strong{font-weight:bolder}code,kbd,samp{font-family:monospace;font-size:1em}small{font-size:80%}sub,sup{vertical-align:baseline;font-size:75%;line-height:0;position:relative}sub{bottom:-.25em}sup{top:-.5em}img{border-style:none}button,input,optgroup,select,textarea{margin:0;font-family:inherit;font-size:100%;line-height:1.15}button,input{overflow:visible}button,select{text-transform:none}button,[type=button],[type=reset],[type=submit]{-webkit-appearance:button;appearance:button}button::-moz-focus-inner,[type=button]::-moz-focus-inner,[type=reset]::-moz-focus-inner,[type=submit]::-moz-focus-inner{border-style:none;padding:0}button:-moz-focusring,[type=button]:-moz-focusring,[type=reset]:-moz-focusring,[type=submit]:-moz-focusring{outline:1px dotted buttontext}fieldset{padding:.35em .75em .625em}legend{color:inherit;box-sizing:border-box;white-space:normal;max-width:100%;padding:0;display:table}progress{vertical-align:baseline}textarea{overflow:auto}[type=checkbox],[type=radio]{box-sizing:border-box;padding:0}[type=number]::-webkit-inner-spin-button,[type=number]::-webkit-outer-spin-button{height:auto}[type=search]{-webkit-appearance:textfield;appearance:textfield;outline-offset:-2px}[type=search]::-webkit-search-decoration{-webkit-appearance:none}::-webkit-file-upload-button{font:inherit;-webkit-appearance:button}details{display:block}summary{display:list-item}template{display:none}html{color:#222;scroll-behavior:smooth;font-size:1em;line-height:1.4}::-moz-selection{text-shadow:none;background:#191919}::selection{text-shadow:none;background:#191919}hr{border:0;border-top:1px solid #ccc;height:1px;margin:1em 0;padding:0;display:block;overflow:visible}audio,canvas,iframe,img,svg,video{vertical-align:middle}fieldset{border:0;margin:0;padding:0}textarea{resize:vertical}body{color:#fafafa;background-color:#191919;margin:10px;font-family:Arial,Helvetica,sans-serif}a{text-decoration:none}a:link{color:#ffc627}a:hover,a:active{color:#ffc627;text-decoration:underline}a:focus{color:#111;background-color:#ffc627}a:visited,a:visited:hover{color:#cba557}a:visited:focus,a:visited:focus-visible{color:#111!important}.hidden,[hidden]{display:none!important}.visually-hidden{clip:rect(0,0,0,0);white-space:nowrap;border:0;width:1px;height:1px;margin:-1px;padding:0;position:absolute;overflow:hidden}.visually-hidden.focusable:active,.visually-hidden.focusable:focus{clip:auto;width:auto;height:auto;white-space:inherit;margin:0;position:static;overflow:visible}.invisible{visibility:hidden}.clearfix:before,.clearfix:after{content:"";display:table}.clearfix:after{clear:both}@media print{*,:before,:after{color:#000!important;box-shadow:none!important;text-shadow:none!important;background:#fff!important}a,a:visited{text-decoration:underline}a[href]:after{content:" (" attr(href)")"}abbr[title]:after{content:" (" attr(title)")"}a[href^=\#]:after,a[href^=javascript\:]:after{content:""}pre{white-space:pre-wrap!important}pre,blockquote{break-inside:avoid;border:1px solid #999}tr,img{break-inside:avoid}p,h2,h3{orphans:3;widows:3}h2,h3{break-after:avoid}}.full-width-section{background-position:50%;background-size:cover;width:100%;max-width:1920px;margin:0 auto}.container{max-width:1200px;margin:0 auto;padding:0 12px}.readable{max-width:900px;margin:0 auto}header,footer{width:100%}header .container,footer .container{max-width:1200px;margin:0 auto;padding:20px 12px}.gh{border-collapse:collapse;border-spacing:0;margin:0 auto}.gh td,.gh th{border-collapse:collapse;word-break:normal;padding:10px 5px;overflow:hidden}.gh .gh-tcell{text-align:center;vertical-align:middle}@media screen and (width<=767px){.gh,.gh col{width:auto!important}.gh-wrap{-webkit-overflow-scrolling:touch;margin:auto 0;overflow-x:auto}}.soc{border-collapse:collapse;border-spacing:0;margin:0 auto}.soc td,.soc th{border-collapse:collapse;word-break:normal;padding:8px;overflow:hidden}.soc .soc-fa{text-align:center;vertical-align:middle}@media screen and (width<=767px){.soc,.soc col{width:auto!important}.soc-wrap{-webkit-overflow-scrolling:touch;margin:auto 0;overflow-x:auto}}.foss{border-collapse:collapse;border-spacing:0}.foss td,.foss th{border-collapse:collapse;word-break:normal;padding:10px 5px;overflow:hidden}.foss .foss-cell{text-align:center;vertical-align:middle}@media screen and (width<=767px){.foss,.foss col{width:auto!important}.foss-wrap{-webkit-overflow-scrolling:touch;overflow-x:auto}}.bnav{text-align:center;border-collapse:collapse;border-spacing:0;margin:0 auto}.bnav td,.bnav th{text-align:center;vertical-align:middle;word-break:normal;border-style:none;padding:10px;font-size:.875rem;font-weight:700;line-height:1.125rem;overflow:hidden}.bnav .bnav-cell{text-align:center;vertical-align:middle;align-content:center}@media screen and (width<=767px){.bnav,.bnav col{width:auto!important}.bnav-wrap{-webkit-overflow-scrolling:touch;margin:auto 0;overflow-x:auto}}.bnav2{border-collapse:collapse;border-spacing:0;margin:0 auto}.bnav2 td{word-break:normal;border-style:none;padding:10px;font-size:.875rem;font-weight:700;line-height:1.125rem;overflow:hidden}.bnav2 th{word-break:normal;border-style:none;padding:12px;font-size:.875rem;line-height:1.125rem;overflow:hidden}.bnav2 .bnav2-cell{text-align:center;vertical-align:middle;align-content:center}@media screen and (width<=767px){.bnav2,.bnav2 col{width:auto!important}.bnav2-wrap{-webkit-overflow-scrolling:touch;margin:auto 0;overflow-x:auto}}.pgp{border-collapse:collapse;border-spacing:0;margin:0 auto}.pgp td{word-break:normal;border-style:none;padding:10px;font-size:.875rem;line-height:1.125rem;overflow:hidden}.pgp th{word-break:normal;border:1px solid #000;padding:10px;font-size:.875rem;line-height:1.125rem;overflow:hidden}.pgp .pgp-col1{text-align:right;vertical-align:middle;padding-right:1rem}.pgp .pgp-col2{text-align:left;vertical-align:middle;padding-left:1rem}@media screen and (width<=767px){.pgp,.pgp col{width:auto!important}.pgp-wrap{-webkit-overflow-scrolling:touch;margin:2rem 0 auto;overflow-x:auto}}#service-summary{color:#e6e6e6;margin-top:2rem;margin-bottom:2.5rem}.service-table{color:#e6e6e6;border-collapse:collapse;background-color:#191919;width:100%;font-size:.95rem}.service-table th,.service-table td{text-align:left;vertical-align:top;border-bottom:1px solid #333;padding:.75rem 1rem}.service-table th{color:#ffc627;text-transform:uppercase;background-color:#222;font-size:.85rem;font-weight:600}.service-table a{color:#ffc627;font-weight:500;text-decoration:none}.service-table a:hover{text-decoration:underline}.service-table tr:hover{background-color:#222;transition:background-color .3s,box-shadow .3s;box-shadow:0 0 10px 2px #ffc62740}.service-table tr.selected{background-color:#222;border-left:4px solid #ffc627;transition:background-color .3s,border-left-color .3s}.service-table tr.selected:hover{background-color:#252525;box-shadow:0 0 12px 3px #ffc62759}@media (width<=768px){.service-table{font-size:.9rem}.service-table th,.service-table td{padding:.5rem}}.logo{margin-left:auto;margin-right:auto;display:block}.index-title1{text-align:center;font-style:italic;font-weight:700}.index-title2{letter-spacing:-.015em;text-align:center;font-variant:small-caps;font-size:1.25rem;line-height:1.625rem}.index1{letter-spacing:-.035em;text-align:center;font-style:italic;font-weight:700;line-height:2.125rem}.index2{letter-spacing:-.035em;text-align:center;font-variant:small-caps;font-size:1.5rem;line-height:1.75rem}.index3{letter-spacing:-.035em;text-align:center;font-size:1.5rem;line-height:1.75rem}.index4{letter-spacing:-.035em;text-align:center;font-size:1.5rem;line-height:1.75rem;text-decoration:underline}.subhead{letter-spacing:-.035em;font-variant:small-caps;font-size:1.5rem;line-height:1.75rem}.bold{font-weight:700}.emphasis{font-style:italic}.uline{text-decoration:underline}.bolditalic{font-style:italic;font-weight:700}.bquote{border-left:3px solid #9e9e9e;margin-left:30px;padding-left:10px;font-style:italic}.small-text{font-size:.75rem;line-height:1.125rem}.large-text-center{text-align:center;font-size:1.25rem;line-height:1.75rem}.prewrap{white-space:pre-wrap;display:block}.hr-styled{width:75%;margin:auto}.center-text{text-align:center}.copyright{text-align:center;font-size:.75rem;line-height:1.125rem}.gold{color:#ffc627}.visited{color:#cba557}.goldseparator{color:#ffc627;margin:0 .5rem}.center-nav{text-align:center;padding:5px;font-size:1rem;line-height:1.5rem}.block{overflow-wrap:break-word;resize:none;white-space:normal;word-break:normal;background:0 0;border:none;border-radius:0;outline:none;width:100%;font-family:monospace;font-size:.875rem;line-height:1.125rem}.full-width-section.centered{flex-direction:column;justify-content:center;min-height:80vh;display:flex}.fingerprint{white-space:pre-line;font-weight:700;display:block}.pgp-image{width:150px;height:150px}.spacer{margin:2rem 0}.separator{margin:0 .5rem}.emoji{margin-right:8px}.headline{margin-bottom:4px;font-style:italic;font-weight:700;display:block}.label{font-family:inherit;font-weight:700}.description{font-family:inherit;font-style:normal;font-weight:400;display:inline}.sr-only{clip:rect(0,0,0,0);white-space:nowrap;border:0;width:1px;height:1px;margin:-1px;padding:0;position:absolute;overflow:hidden}.pgp-entry{flex-wrap:wrap;align-items:center;gap:2rem;margin-bottom:2rem;display:flex}.pgp-text{flex:2;min-width:250px}.pgp-qr{flex:1;min-width:150px}.obtainium-direct-label{margin:.25rem 0 .75rem;font-weight:700}.obtainium-manual-label{margin-top:.75rem;font-weight:700}.obtainium-img{width:185px;height:55px;margin-bottom:.25rem}.obtainium-margin{margin-left:4px}.obtainium-fa-down{color:#ffc627;margin-left:4px}.obtainium-icon{width:50px;height:50px}.proton-img{width:168px;height:24px}.redirect-text{text-align:center;margin-top:4rem}.redirect-content{text-align:center;margin-top:2rem;font-family:system-ui,sans-serif}.loading-spinner{border:4px solid #ddd;border-top-color:#ffc627;border-radius:50%;width:48px;height:48px;margin:2rem auto;animation:1s linear infinite spin}@keyframes spin{to{transform:rotate(360deg)}}.cc-link{text-decoration:none}.cc-img{vertical-align:text-bottom;margin-left:3px;display:inline-block;height:18px!important}#toc ul{padding-left:1.5rem;list-style-type:disc}#toc a{color:var(--color-primary,#ffc627);text-decoration:none}#toc a:hover{text-decoration:underline}
@@ -9,4 +9,3 @@ This file is part of Network Pro.
9
9
  // Import transformed and minified stylesheets
10
10
  import "./global.min.css";
11
11
  import "./fa-global.css";
12
-
@@ -0,0 +1,50 @@
1
+ <!-- =====================================================================
2
+ src/lib/types/README.md
3
+
4
+ Copyright © 2025 Network Pro Strategies (Network Pro™)
5
+ SPDX-License-Identifier: CC-BY-4.0 OR GPL-3.0-or-later
6
+ This file is part of Network Pro.
7
+ ====================================================================== -->
8
+
9
+ # Typing Policy
10
+
11
+ This directory contains JSDoc-based type definitions for reusable modules
12
+ within the Network Pro codebase. The goal is to provide meaningful IntelliSense,
13
+ editor documentation, and lightweight type safety without requiring TypeScript
14
+ conversion.
15
+
16
+ ## Scope and Philosophy
17
+
18
+ - **Structured Data Only** – Typing is focused on objects that benefit from
19
+ explicit schemas (e.g., configuration, constants, utilities).
20
+ - **Svelte Components** – Components are intentionally _not_ fully typed at this
21
+ stage. Component auto-completion is already handled by the Svelte language
22
+ server, and explicit typedefs would add unnecessary complexity.
23
+ - **Scalability** – Typing is modular. New typedefs can be added under
24
+ `/src/lib/types/` as the codebase grows.
25
+ - **Documentation-Driven** – JSDoc comments serve both as type definitions and
26
+ developer documentation. This minimizes friction while improving clarity.
27
+
28
+ ## Example
29
+
30
+ ```js
31
+ /** @typedef {import('./types/appConstants.js').AppConstants} AppConstants */
32
+ /** @type {AppConstants} */
33
+ export const CONSTANTS = { ... };
34
+ ```
35
+
36
+ This approach balances clarity and maintainability without introducing a
37
+ TypeScript build requirement.
38
+
39
+ ---
40
+
41
+ <span style="font-size: 12px; text-align: center;">
42
+
43
+ Copyright &copy; 2025
44
+ **[Network Pro Strategies](https://netwk.pro) (Network Pro&trade;)**
45
+
46
+ Network Pro&trade;, the shield logo, and the "Locking Down Networks...&trade;" slogan are [trademarks](https://netwk.pro/license#trademark) of Network Pro Strategies.
47
+
48
+ Licensed under **[CC BY 4.0](https://netwk.pro/license#cc-by)** and the **[GNU GPL](https://netwk.pro/license#gnu-gpl)**, as published by the [Free Software Foundation](https://www.fsf.org), either version 3 of the License, or (at your option) any later version.
49
+
50
+ </span>
@@ -10,7 +10,7 @@ This file is part of Network Pro.
10
10
  * @file appConstants.js
11
11
  * @description Type definitions for app constants in src/lib/index.js
12
12
  * @module src/lib/types
13
- * @author SunDevil311
13
+ * @author Scott Lopez
14
14
  * @updated 2025-06-03
15
15
  */
16
16
 
@@ -11,7 +11,7 @@ This file is part of Network Pro.
11
11
  * @description Type definitions for `fossItem` in
12
12
  * src/lib/components/foss/FossItemContent.svelte
13
13
  * @module src/lib/types
14
- * @author SunDevil311
14
+ * @author Scott Lopez
15
15
  * @updated 2025-06-12
16
16
  */
17
17
 
@@ -12,7 +12,7 @@ This file is part of Network Pro.
12
12
  * Useful for analytics platforms like PostHog or GA4.
13
13
  *
14
14
  * @module src/lib/utils/
15
- * @author SunDevil311
15
+ * @author Scott Lopez
16
16
  * @updated 2025-06-30
17
17
  */
18
18
 
@@ -11,7 +11,7 @@ This file is part of Network Pro.
11
11
  * @description Universal DOMPurify instance for SSR + client with safe build support.
12
12
  * Secures untrusted HTML before injecting it into the DOM.
13
13
  * @module src/lib/utils/
14
- * @author SunDevil311
14
+ * @author Scott Lopez
15
15
  * @updated 2025-06-01
16
16
  */
17
17
 
@@ -12,7 +12,7 @@ This file is part of Network Pro.
12
12
  * visual if true.
13
13
  *
14
14
  * @module src/lib/utils/
15
- * @author SunDevil311
15
+ * @author Scott Lopez
16
16
  * @updated 2025-07-01
17
17
  */
18
18
 
@@ -10,8 +10,8 @@ This file is part of Network Pro.
10
10
  * @file utm.js
11
11
  * @description Appends standardized UTM parameters to a given URL.
12
12
  * @module src/lib/utils/
13
- * @author SunDevil311
14
- * @updated 2025-06-30
13
+ * @author Scott Lopez
14
+ * @updated 2025-10-10
15
15
  */
16
16
 
17
17
  import { browser } from '$app/environment';
@@ -14,21 +14,13 @@ This file is part of Network Pro.
14
14
  import { initPostHog, showReminder, capture } from '$lib/stores/posthog';
15
15
  import { registerServiceWorker } from '$lib/registerServiceWorker.js';
16
16
  import { browser } from '$app/environment';
17
-
18
- import ContainerSection from '$lib/components/ContainerSection.svelte';
19
- import Footer from '$lib/components/layout/Footer.svelte';
20
- import HeaderDefault from '$lib/components/layout/HeaderDefault.svelte';
21
- import HeaderHome from '$lib/components/layout/HeaderHome.svelte';
22
- import PWAInstallButton from '$lib/components/PWAInstallButton.svelte';
17
+ import { logoPng, logoWbp, faviconSvg, appleTouchIcon } from '$lib';
18
+ import { ContainerSection, PWAInstallButton } from '$lib/components';
19
+ import { Footer, HeaderDefault, HeaderHome } from '$lib/components/layout';
23
20
 
24
21
  import '$lib/styles/global.min.css';
25
22
  import '$lib/styles/fa-global.css';
26
23
 
27
- import logoPng from '$lib/img/logo-web.png';
28
- import logoWbp from '$lib/img/logo-web.webp';
29
- import faviconSvg from '$lib/img/favicon.svg';
30
- import appleTouchIcon from '$lib/img/icon-180x180.png';
31
-
32
24
  $: shouldShowReminder = $showReminder;
33
25
 
34
26
  onMount(() => {
@@ -7,13 +7,15 @@ This file is part of Network Pro.
7
7
  ========================================================================== -->
8
8
 
9
9
  <script>
10
- import Badges from '$lib/components/Badges.svelte';
11
- import FullWidthSection from '$lib/components/FullWidthSection.svelte';
12
- import HomeContent from '$lib/pages/HomeContent.svelte';
13
- import LegalNav from '$lib/components/LegalNav.svelte';
14
- import Logo from '$lib/components/Logo.svelte';
15
- import SocialMedia from '$lib/components/SocialMedia.svelte';
16
- import MetaTags from '$lib/components/MetaTags.svelte';
10
+ import {
11
+ Badges,
12
+ FullWidthSection,
13
+ LegalNav,
14
+ Logo,
15
+ SocialMedia,
16
+ MetaTags,
17
+ } from '$lib/components';
18
+ import { HomeContent } from '$lib/pages';
17
19
 
18
20
  /**
19
21
  * @type {string}
@@ -7,7 +7,7 @@ This file is part of Network Pro.
7
7
  ========================================================================== -->
8
8
 
9
9
  <script>
10
- import RedirectPage from '$lib/components/RedirectPage.svelte';
10
+ import { RedirectPage } from '$lib/components';
11
11
  import { appendUTM } from '$lib/utils/utm.js';
12
12
  import { getUTMParams } from '$lib/utils/getUTMParams.js';
13
13
  import { trackingEnabled } from '$lib/stores/trackingPreferences';
@@ -7,16 +7,17 @@ This file is part of Network Pro.
7
7
  ========================================================================== -->
8
8
 
9
9
  <script>
10
- import RedirectPage from '$lib/components/RedirectPage.svelte';
11
10
  import { appendUTM } from '$lib/utils/utm.js';
12
11
  import { getUTMParams } from '$lib/utils/getUTMParams.js';
13
12
  import { trackingEnabled } from '$lib/stores/trackingPreferences';
14
- import posthog from 'posthog-js';
13
+ import { RedirectPage } from '$lib/components';
15
14
  import { get } from 'svelte/store';
16
15
  import { onMount } from 'svelte';
17
16
  import { browser } from '$app/environment';
18
17
  import { CONSTANTS } from '$lib';
19
18
 
19
+ import posthog from 'posthog-js';
20
+
20
21
  //console.log(CONSTANTS.COMPANY_INFO.APP_NAME);
21
22
 
22
23
  const { PAGE } = CONSTANTS;
@@ -0,0 +1,18 @@
1
+ /* ==========================================================================
2
+ src/routes/services/+page.server.js
3
+
4
+ Copyright © 2025 Network Pro Strategies (Network Pro™)
5
+ SPDX-License-Identifier: CC-BY-4.0 OR GPL-3.0-or-later
6
+ This file is part of Network Pro.
7
+ ========================================================================== */
8
+
9
+ import { defaultMeta, meta } from '$lib/meta.js';
10
+
11
+ export const prerender = false;
12
+
13
+ /** @type {import('./$types').PageServerLoad} */
14
+ export function load() {
15
+ return {
16
+ meta: meta['/services'] || defaultMeta,
17
+ };
18
+ }
@@ -0,0 +1,65 @@
1
+ <!-- ==========================================================================
2
+ src/routes/services/+page.svelte
3
+
4
+ Copyright © 2025 Network Pro Strategies (Network Pro™)
5
+ SPDX-License-Identifier: CC-BY-4.0 OR GPL-3.0-or-later
6
+ This file is part of Network Pro.
7
+ ========================================================================== -->
8
+
9
+ <script>
10
+ import {
11
+ Badges,
12
+ FullWidthSection,
13
+ LegalNav,
14
+ SocialMedia,
15
+ MetaTags,
16
+ } from '$lib/components';
17
+ import { ServicesContent } from '$lib/pages';
18
+
19
+ /**
20
+ * @type {string}
21
+ * Style class for the div element.
22
+ */
23
+ const spaceStyle = 'spacer';
24
+
25
+ export let data;
26
+ </script>
27
+
28
+ <MetaTags title={data.meta.title} description={data.meta.description} />
29
+
30
+ <link rel="canonical" href="https://netwk.pro/services" />
31
+
32
+ <section id="services">
33
+ <FullWidthSection>
34
+ <!-- BEGIN PRIVACY POLICY -->
35
+ <section id="main-content">
36
+ <ServicesContent />
37
+ </section>
38
+ <!-- END PRIVACY POLICY -->
39
+
40
+ <div class={spaceStyle}></div>
41
+
42
+ <!-- BEGIN SOCIAL MEDIA -->
43
+ <section id="social-media">
44
+ <SocialMedia />
45
+ </section>
46
+ <!-- END SOCIAL MEDIA -->
47
+
48
+ <hr />
49
+
50
+ <div class={spaceStyle}></div>
51
+
52
+ <!-- BEGIN LEGAL -->
53
+ <section id="legal-nav">
54
+ <LegalNav />
55
+ </section>
56
+ <!-- END LEGAL -->
57
+
58
+ <div class={spaceStyle}></div>
59
+
60
+ <!-- BEGIN BADGES -->
61
+ <section id="badges">
62
+ <Badges />
63
+ </section>
64
+ </FullWidthSection>
65
+ </section>
@@ -15,7 +15,7 @@ This file is part of Network Pro.
15
15
  * injected and available during runtime in Netlify's SSR context.
16
16
  *
17
17
  * @module src/routes/status
18
- * @author SunDevil311
18
+ * @author Scott Lopez
19
19
  * @updated 2025-05-31
20
20
  */
21
21
 
@@ -39,8 +39,6 @@ const IGNORE_PATHS = new Set([
39
39
  '/pgp/github@sl.neteng.cc.asc',
40
40
  '/pgp/security@s.neteng.pro.asc',
41
41
  '/pgp/support@neteng.pro.asc',
42
- '/pgp/vcard.png',
43
- '/pgp/vcard.webp',
44
42
  '/screenshots/desktop-foss.png',
45
43
  '/webfonts/fa-brands-400.ttf',
46
44
  '/webfonts/fa-solid-900.ttf',
@@ -96,6 +94,8 @@ const REQUIRED_ASSETS = [
96
94
  '/pgp/pgp-security.webp',
97
95
  '/pgp/pgp-support.png',
98
96
  '/pgp/pgp-support.webp',
97
+ '/pgp/vcard.png',
98
+ '/pgp/vcard.webp',
99
99
  '/manifest.json',
100
100
  '/offline.html',
101
101
  '/offline.min.css',
@@ -3,11 +3,10 @@ VERSION:3.0
3
3
  FN:Scott Lopez
4
4
  N:Lopez;Scott
5
5
  TEL;TYPE=WORK,PREF=1:(623) 252-4350
6
- EMAIL;TYPE=HOME:github@sl.neteng.cc
7
6
  EMAIL;TYPE=WORK:support@neteng.pro
8
7
  EMAIL;TYPE=Secure:business@s.neteng.pro
9
8
  ADR;TYPE=WORK:;;Peoria\, AZ 85382\nUS
10
9
  ORG:Network Pro Strategies
11
- TITLE:Security Consultant
10
+ TITLE:Network Engineer
12
11
  URL:https://netwk.pro
13
12
  END:VCARD
@@ -10,7 +10,7 @@ This file is part of Network Pro.
10
10
  * @file disableSw.js
11
11
  * @description Allows for Service Worker to be disabled for debugging by appending ?nosw to the path.
12
12
  * @module static
13
- * @author SunDevil311
13
+ * @author Scott Lopez
14
14
  * @updated 2025-10-05
15
15
  */
16
16
 
@@ -1,5 +1,5 @@
1
1
  <?xml version="1.0" encoding="UTF-8"?>
2
- <!-- Sitemap last updated 2025-09-27 -->
2
+ <!-- Sitemap last updated 2025-10-17 -->
3
3
 
4
4
  <urlset xmlns="http://www.sitemaps.org/schemas/sitemap/0.9">
5
5
 
@@ -7,7 +7,7 @@
7
7
 
8
8
  <loc>https://netwk.pro</loc>
9
9
 
10
- <lastmod>2025-09-27</lastmod>
10
+ <lastmod>2025-10-17</lastmod>
11
11
 
12
12
  <changefreq>weekly</changefreq>
13
13
 
@@ -15,6 +15,18 @@
15
15
 
16
16
  </url>
17
17
 
18
+ <url>
19
+
20
+ <loc>https://netwk.pro/services</loc>
21
+
22
+ <lastmod>2025-10-17</lastmod>
23
+
24
+ <changefreq>monthly</changefreq>
25
+
26
+ <priority>0.8</priority>
27
+
28
+ </url>
29
+
18
30
  <url>
19
31
 
20
32
  <loc>https://netwk.pro/foss-spotlight</loc>
@@ -31,7 +43,7 @@
31
43
 
32
44
  <loc>https://netwk.pro/about</loc>
33
45
 
34
- <lastmod>2025-06-10</lastmod>
46
+ <lastmod>2025-10-17</lastmod>
35
47
 
36
48
  <changefreq>monthly</changefreq>
37
49
 
@@ -55,7 +67,7 @@
55
67
 
56
68
  <loc>https://netwk.pro/privacy</loc>
57
69
 
58
- <lastmod>2025-06-30</lastmod>
70
+ <lastmod>2025-10-17</lastmod>
59
71
 
60
72
  <changefreq>monthly</changefreq>
61
73
 
@@ -67,7 +79,7 @@
67
79
 
68
80
  <loc>https://netwk.pro/license</loc>
69
81
 
70
- <lastmod>2025-06-10</lastmod>
82
+ <lastmod>2025-10-09</lastmod>
71
83
 
72
84
  <changefreq>monthly</changefreq>
73
85
 
@@ -10,7 +10,7 @@ This file is part of Network Pro.
10
10
  * @file app.spec.js
11
11
  * @description Runs Playwright E2E tests with desktop and root route assertions.
12
12
  * @module tests/e2e
13
- * @author SunDevil311
13
+ * @author Scott Lopez
14
14
  * @updated 2025-09-17
15
15
  */
16
16
 
@@ -10,7 +10,7 @@ This file is part of Network Pro.
10
10
  * @file mobile.spec.js
11
11
  * @description Runs Playwright E2E tests with mobile assertions.
12
12
  * @module tests/e2e
13
- * @author SunDevil311
13
+ * @author Scott Lopez
14
14
  * @updated 2025-09-17
15
15
  */
16
16
 
@@ -10,7 +10,7 @@ This file is part of Network Pro.
10
10
  * @file helpers.js
11
11
  * @description Stores commonly used functions for importing into E2E tests.
12
12
  * @module tests/e2e/shared
13
- * @author SunDevil311
13
+ * @author Scott Lopez
14
14
  * @updated 2025-05-29
15
15
  */
16
16
 
@@ -10,7 +10,7 @@ This file is part of Network Pro.
10
10
  * @file meta.test.js
11
11
  * @description Checks for correct metadata population in CI
12
12
  * @module tests/meta
13
- * @author SunDevil311
13
+ * @author Scott Lopez
14
14
  * @updated 2025-09-17
15
15
  */
16
16