Skip to content

Commit

Permalink
feat: CrUX tab
Browse files Browse the repository at this point in the history
  • Loading branch information
harlan-zw committed Mar 3, 2024
1 parent 6971170 commit b2923fa
Show file tree
Hide file tree
Showing 17 changed files with 998 additions and 103 deletions.
2 changes: 1 addition & 1 deletion docs/composables/shiki.ts
Original file line number Diff line number Diff line change
@@ -1,4 +1,4 @@
import type { Highlighter, BundledLanguage } from 'shiki'
import type { BundledLanguage, Highlighter } from 'shiki'
import { getHighlighter } from 'shiki'
import { computed, ref, unref } from 'vue'
import type { MaybeRef } from '@vueuse/core'
Expand Down
2 changes: 0 additions & 2 deletions package.json
Original file line number Diff line number Diff line change
Expand Up @@ -12,9 +12,7 @@
"build:docs": "cd docs && pnpm i && nuxi build",
"build:pkg": "pnpm -r --filter=./packages/** run build",
"stub": "JITI_ESM_RESOLVE=true && pnpm -r --parallel run stub",
"dev": "pnpm run stub && pnpm run watch:client",
"lint": "eslint . --fix",
"lint:fix": "npm run lint -- --fix",
"release": "bumpp package.json packages/*/package.json --commit --push --tag",
"test": "vitest",
"test:update": "vitest -u",
Expand Down
265 changes: 186 additions & 79 deletions packages/client/App.vue
Original file line number Diff line number Diff line change
@@ -1,6 +1,8 @@
<script setup lang="ts">
import { useTitle } from '@vueuse/core'
import { $fetch } from 'ofetch'
import {
activeTab,
apiUrl,
basePath,
categoryScores,
Expand Down Expand Up @@ -29,12 +31,18 @@ import {
wsConnect,
} from './logic'
const crux = ref(null)
if (!isStatic) {
onMounted(() => {
wsConnect()
setInterval(() => {
refreshScanMeta()
}, 5000)
$fetch(`https://crux.unlighthouse.dev/api/${encodeURIComponent(website)}/crux/history`).then((res) => {
crux.value = res
})
})
}
Expand All @@ -45,9 +53,9 @@ useTitle(`${website.replace(/https?:\/\/(www.)?/, '')} | Unlighthouse`)
<main class="text-gray-700 dark:text-gray-200 overflow-y-hidden max-h-screen h-screen grid grid-rows-[min-content,1fr]">
<NavBar />
<div class="2xl:flex mt-2">
<div class="xl:ml-3 mx-3 mr-0 w-full 2xl:(mr-5 w-250px mb-0) mb-3">
<div class="flex justify-between max-h-[95%] flex-col xl:ml-3 mx-3 mr-0 w-full 2xl:(mr-5 w-250px mb-0)">
<TabGroup vertical @change="changedTab">
<TabList class="p-1 dark:(bg-blue-900/20 border-none) border-2 border-blue-900/30 rounded-xl 2xl:(mt-8 block) flex">
<TabList class="2xl:(block) flex">
<Tab
v-for="(category, key) in tabs"
:key="key"
Expand All @@ -57,21 +65,24 @@ useTitle(`${website.replace(/https?:\/\/(www.)?/, '')} | Unlighthouse`)
<btn-tab
:selected="selected"
>
<span class="inline-flex items-center">{{ category }}
<tooltip v-if="category === 'Performance'" class="text-left">
<span class="inline-flex items-center">
<component :is="category.icon" class="inline text-sm opacity-40 h-4 w-4 mr-2" />
<span>{{ category.label }}</span>
<tooltip v-if="category.label === 'Performance'" class="text-left">
<i-carbon-warning class="inline text-xs mx-1" />
<template #tooltip>
<div class="mb-2">Lighthouse is running with variability. Performance scores should not be considered accurate.</div>
<div>Unlighthouse is running <span class="underline">with{{ throttle ? '' : 'out' }} throttling</span> which will also effect scores.</div>
</template>
</tooltip>
</span>
<metric-guage v-if="category !== 'Overview' && !Number.isNaN(categoryScores[key - 1])" :score="categoryScores[key - 1]" :stripped="true" class="dark:font-bold" :class="selected ? ['dark:bg-teal-900 bg-blue-100 rounded px-2'] : []" />
<metric-guage v-if="!['Overview', 'CrUX'].includes(category.label) && !Number.isNaN(categoryScores[key - 1])" :score="categoryScores[key - 1]" :stripped="true" class="dark:font-bold" :class="selected ? ['dark:bg-teal-900 bg-blue-100 rounded px-2'] : []" />
</btn-tab>
</Tab>
</TabList>
</TabGroup>
<div class="hidden 2xl:block">
<lighthouse-three-d v-if="!isStatic" class="mb-7" />
<div class="px-2 text-center 2xl:text-left">
<div class="text-xs opacity-75 2xl:mt-4">
<a href="https://unlighthouse.dev" target="_blank" class="underline hover:no-underline">Documentation</a>
Expand All @@ -86,90 +97,186 @@ useTitle(`${website.replace(/https?:\/\/(www.)?/, '')} | Unlighthouse`)
Portions of this report use Lighthouse. For more information visit <a href="https://developers.google.com/web/tools/lighthouse" class="underline hover:no-underline">here</a>.
</div>
</div>
<lighthouse-three-d v-if="!isStatic" />
</div>
</div>
<div class="xl:w-full w-screen overflow-x-auto px-3">
<div class="pr-10 py-1 w-full min-w-1500px">
<div class="grid grid-cols-12 gap-4 text-sm dark:(text-gray-300) text-gray-700">
<results-table-head
v-for="(column, key) in resultColumns"
:key="key"
:sorting="sorting"
:column="column"
@sort="incrementSort"
/>
<div v-if="tabs[activeTab].label === 'CrUX'">
<div>
<h2 class="font-bold text-2xl mb-7">
Origin CrUX History - Mobile
</h2>
</div>
</div>
<div class="w-full min-w-1500px 2xl:(max-h-[calc(100vh-100px)]) lg:max-h-[calc(100vh-205px)] sm:max-h-[calc(100vh-220px)] max-h-[calc(100vh-250px)] overflow-auto pr-5 mr-4">
<div v-if="Object.values(searchResults).length === 0" class="px-4 py-3">
<template v-if="searchText">
<p class="mb-2">
No results for search "{{ searchText }}"...
</p>
<btn-action class="dark:(bg-teal-700) bg-blue-100 px-2 text-sm" @click="searchText = ''">
Reset Search
</btn-action>
</template>
<div v-else class="flex items-center">
<loading-spinner class="mr-2" />
<div v-if="!crux" class="w-full">
<div class="text-gray-500 text-center w-full text-sm">
Loading CrUX data...
</div>
</div>
<div v-else-if="crux?.exists === false" class="w-full">
<div class="text-gray-500 text-center w-full text-sm">
No data from Chrome UX report
</div>
</div>
<div v-else class="w-full flex-col flex space-y-5">
<div>
<div>
<a href="https://web.dev/articles/inp" target="_blank" class="transition hover:underline">Interaction to Next Paint (INP)</a>
</div>
<div v-if="crux?.inp" class="flex items-center">
<div class="w-full h-[200px] w-[400px] relative">
<CruxGraphInp v-if="crux?.inp" :value="crux.inp" :height="200" />
</div>
<div>
<div class="text-green-500 font-bold">
Good &lt; 200ms
</div>
<div class="text-yellow-500 font-bold">
Needs Improvement 200ms - 500ms
</div>
<div class="text-red-500 font-bold">
Poor &gt; 500ms
</div>
</div>
</div>
<div v-else class="inline">
<div class="inline text-gray-500 text-center w-full text-sm">
No data
</div>
</div>
</div>
<div>
<div><a href="https://web.dev/articles/cls" target="_blank" class="transition hover:underline">Cumulative Layout Shift (CLS)</a></div>
<div v-if="crux?.cls" class="flex items-center">
<div class="w-full h-[200px] w-[400px] relative">
<CruxGraphCls v-if="crux?.cls" :value="crux.cls" :height="200" />
</div>
<div>
<div class="text-green-500 font-bold">
Good &lt; 0.1
</div>
<div class="text-yellow-500 font-bold">
Needs Improvement 0.1 - 0.25
</div>
<div class="text-red-500 font-bold">
Poor &gt; 0.25
</div>
</div>
</div>
<div v-else class="inline">
<div class="text-gray-500 inline text-center w-full text-sm">
No data
</div>
</div>
</div>
<div>
<div>
<p>Waiting for routes...</p>
<span class="text-xs opacity-50">If this hangs consider running Unlighthouse with --debug.</span>
<a href="https://web.dev/articles/lcp" target="_blank" class="transition hover:underline">Largest Contentful Paint (LCP)</a>
</div>
<div v-if="crux?.lcp" class="flex items-center">
<div class="w-full h-[200px] w-[400px] relative">
<CruxGraphLcp v-if="crux?.lcp" :value="crux.lcp" :height="200" />
</div>
<div>
<div class="text-green-500 font-bold">
Good &lt; 2.5s
</div>
<div class="text-yellow-500 font-bold">
Needs Improvement 2.5s - 4s
</div>
<div class="text-red-500 font-bold">
Poor &gt; 4s
</div>
</div>
</div>
<div v-else class="inline">
<div class="text-gray-500 text-center inline w-full text-sm">
No data
</div>
</div>
</div>
</div>
<div v-else-if="searchText" class="px-4 py-3">
<p>Showing {{ Object.values(searchResults).flat().length }} routes for search "{{ searchText }}":</p>
</div>
<template v-else>
<div class="pr-10 pb-1 w-full min-w-1500px">
<div class="grid grid-cols-12 gap-4 text-sm dark:(text-gray-300) text-gray-700">
<results-table-head
v-for="(column, key) in resultColumns"
:key="key"
:sorting="sorting"
:column="column"
@sort="incrementSort"
/>
</div>
</div>
<results-row
v-for="(reports, routeName) in searchResults"
:key="routeName"
:reports="reports"
:route-name="routeName"
>
<template #actions="{ report }">
<popover-actions position="left">
<div class="w-300px flex flex-col space-y-2">
<btn-basic
v-if="report.report"
class="flex items-start hover:bg-blue-500 transition children:hover:text-white"
@click="openLighthouseReportIframeModal(report)"
>
<div style="flex-basis: 70px;" class="mt-1 text-blue-500">
<i-vscode-icons-file-type-lighthouse class="text-xl mr-2" />
</div>
<div class="text-left">
<p class="break-none text-base">
Open Lighthouse Report
</p>
<span class="opacity-70 text-xs">
Lighthouse HTML report is opened in a modal.
</span>
</div>
</btn-basic>
<btn-basic
:disabled="isOffline ? 'disabled' : false"
class="flex items-start hover:bg-blue-500 transition children:hover:text-white"
@click="rescanRoute(report.route)"
>
<div style="flex-basis: 70px;" class="mt-1 text-blue-500">
<i-mdi-magnify-scan class="text-xl" />
</div>
<div class="text-left">
<p class="break-none text-base">
Rescan Route
</p>
<span class="opacity-70 text-xs">
Crawl the route again and generate a fresh report.
</span>
</div>
</btn-basic>
<div class="w-full min-w-1500px 2xl:(max-h-[calc(100vh-100px)]) lg:max-h-[calc(100vh-205px)] sm:max-h-[calc(100vh-220px)] max-h-[calc(100vh-250px)] overflow-auto pr-5 mr-4">
<div v-if="Object.values(searchResults).length === 0" class="px-4 py-3">
<template v-if="searchText">
<p class="mb-2">
No results for search "{{ searchText }}"...
</p>
<btn-action class="dark:(bg-teal-700) bg-blue-100 px-2 text-sm" @click="searchText = ''">
Reset Search
</btn-action>
</template>
<div v-else class="flex items-center">
<loading-spinner class="mr-2" />
<div>
<p>Waiting for routes...</p>
<span class="text-xs opacity-50">If this hangs consider running Unlighthouse with --debug.</span>
</div>
</popover-actions>
</template>
</results-row>
</div>
</div>
</div>
<div v-else-if="searchText" class="px-4 py-3">
<p>Showing {{ Object.values(searchResults).flat().length }} routes for search "{{ searchText }}":</p>
</div>
<results-row
v-for="(reports, routeName) in searchResults"
:key="routeName"
:reports="reports"
:route-name="routeName"
>
<template #actions="{ report }">
<popover-actions position="left">
<div class="w-300px flex flex-col space-y-2">
<btn-basic
v-if="report.report"
class="flex items-start hover:bg-blue-500 transition children:hover:text-white"
@click="openLighthouseReportIframeModal(report)"
>
<div style="flex-basis: 70px;" class="mt-1 text-blue-500">
<i-vscode-icons-file-type-lighthouse class="text-xl mr-2" />
</div>
<div class="text-left">
<p class="break-none text-base">
Open Lighthouse Report
</p>
<span class="opacity-70 text-xs">
Lighthouse HTML report is opened in a modal.
</span>
</div>
</btn-basic>
<btn-basic
:disabled="isOffline ? 'disabled' : false"
class="flex items-start hover:bg-blue-500 transition children:hover:text-white"
@click="rescanRoute(report.route)"
>
<div style="flex-basis: 70px;" class="mt-1 text-blue-500">
<i-mdi-magnify-scan class="text-xl" />
</div>
<div class="text-left">
<p class="break-none text-base">
Rescan Route
</p>
<span class="opacity-70 text-xs">
Crawl the route again and generate a fresh report.
</span>
</div>
</btn-basic>
</div>
</popover-actions>
</template>
</results-row>
</div>
</template>
</div>
</div>
<footer class="block 2xl:hidden my-2">
Expand Down
4 changes: 3 additions & 1 deletion packages/client/components.d.ts
Original file line number Diff line number Diff line change
Expand Up @@ -34,14 +34,16 @@ declare module 'vue' {
CellTapTargets: typeof import('./components/Cell/CellTapTargets.vue')['default']
CellWebVitals: typeof import('./components/Cell/CellWebVitals.vue')['default']
Container: typeof import('./components/Container.vue')['default']
CruxGraphCls: typeof import('./components/Crux/Graph/CruxGraphCls.vue')['default']
CruxGraphInp: typeof import('./components/Crux/Graph/CruxGraphInp.vue')['default']
CruxGraphLcp: typeof import('./components/Crux/Graph/CruxGraphLcp.vue')['default']
Dialog: typeof import('@headlessui/vue')['Dialog']
DialogOverlay: typeof import('@headlessui/vue')['DialogOverlay']
Disclosure: typeof import('@headlessui/vue')['Disclosure']
DisclosureButton: typeof import('@headlessui/vue')['DisclosureButton']
DisclosureHandle: typeof import('./components/Disclosure/DisclosureHandle.vue')['default']
DisclosurePanel: typeof import('@headlessui/vue')['DisclosurePanel']
ErrorChip: typeof import('./components/Chip/ErrorChip.vue')['default']
ICarbonCaretRight: typeof import('~icons/carbon/caret-right')['default']
ICarbonChevronSort: typeof import('~icons/carbon/chevron-sort')['default']
ICarbonChevronSortDown: typeof import('~icons/carbon/chevron-sort-down')['default']
ICarbonChevronSortUp: typeof import('~icons/carbon/chevron-sort-up')['default']
Expand Down
5 changes: 2 additions & 3 deletions packages/client/components/Btn/BtnTab.vue
Original file line number Diff line number Diff line change
Expand Up @@ -5,10 +5,9 @@ defineProps<{
</script>

<template>
<div class="flex items-center 2xl:(mb-1) focus:outline-none justify-between">
<i-carbon-caret-right v-if="selected" class="inline text-sm opacity-80 mr-1" />
<div class="flex items-center focus:outline-none justify-between">
<button
class="focus:(outline-none ring-1) ring-offset-blue-400 ring-white ring-opacity-60 flex flex-col px-1 py-1 text-xs lg:(px-2 flex-row py-2 text-sm) items-center justify-between w-full leading-5 transition font-medium xl:text-base text-blue-700 rounded-lg"
class="focus:(outline-none ring-1) ring-offset-blue-400 ring-white ring-opacity-60 flex flex-col text-xs lg:(px-2 flex-row py-1 text-sm) items-center justify-between w-full leading-5 transition font-medium text-blue-700 rounded-lg"
:class="[
selected
? 'bg-blue-900 text-blue-200 shadow'
Expand Down
4 changes: 2 additions & 2 deletions packages/client/components/Cell/CellNetworkRequests.vue
Original file line number Diff line number Diff line change
Expand Up @@ -36,8 +36,8 @@ const requestsMapped = computed(() => {
<template>
<div v-if="value" class="text-sm">
<div class="opacity-90 flex items-center mb-1">
<span>{{ value.details.items.length }} total</span>
<span class="opacity-70 ml-1">{{ totalTransfer }}</span>
<span>{{ totalTransfer }}</span>
<span class="opacity-70 ml-1">{{ value.details.items.length }} total</span>
</div>
<div class="grid gap-2 grid-cols-2">
<div v-for="(group, resourceType) in requestsMapped" :key="resourceType" class="text-xs flex items-center">
Expand Down

0 comments on commit b2923fa

Please sign in to comment.