chore: added more data to the applications

This commit is contained in:
Andre Henriques 2025-04-01 12:58:11 +01:00
parent 1b6a887648
commit 141cfbf7a6
10 changed files with 220 additions and 46 deletions

View File

@ -52,6 +52,8 @@ data class Application(
var create_time: String,
var simple_url: String,
var job_level: String,
var inperson_type: String,
var location: String,
var flairs: List<Flair>,
var events: List<Event>,
var urls: List<String>,
@ -73,6 +75,8 @@ data class Application(
rs.getString("create_time"),
rs.getString("simple_url"),
rs.getString("job_level"),
rs.getString("inperson_type"),
rs.getString("location"),
emptyList(),
emptyList(),
emptyList(),
@ -155,6 +159,8 @@ class ApplicationsController(
"",
"",
"",
"",
"",
emptyList(),
emptyList(),
emptyList(),
@ -511,6 +517,8 @@ class ApplicationService(
"",
if (elm.contains("linkedin")) elm.split("?")[0] else "",
"",
"",
"",
emptyList(),
emptyList(),
emptyList(),
@ -533,7 +541,7 @@ class ApplicationService(
// Create time is auto created by the database
db.update(
"insert into applications (id, url, title, user_id, extra_data, payrange, status_id, company, recruiter, message, agency, simple_url, job_level) values (?,?,?,?,?,?,?,?,?,?,?,?,?);",
"insert into applications (id, url, title, user_id, extra_data, payrange, status_id, company, recruiter, message, agency, simple_url, job_level, inperson_type, location) values (?,?,?,?,?,?,?,?,?,?,?,?,?,?,?);",
application.id,
application.url,
application.title,
@ -547,6 +555,8 @@ class ApplicationService(
application.agency,
application.simple_url,
application.job_level,
application.inperson_type,
application.location,
)
eventService.create(application.id, EventType.Creation)
@ -594,7 +604,7 @@ class ApplicationService(
public fun update(application: Application): Application {
// I don't want ot update create_time
db.update(
"update applications set url=?, title=?, user_id=?, extra_data=?, payrange=?, company=?, recruiter=?, message=?, agency=?, simple_url=?, job_level=? where id=?",
"update applications set url=?, title=?, user_id=?, extra_data=?, payrange=?, company=?, recruiter=?, message=?, agency=?, simple_url=?, job_level=?, inperson_type=?, location=? where id=?",
application.url,
application.title,
application.user_id,
@ -606,6 +616,8 @@ class ApplicationService(
application.agency,
application.simple_url,
application.job_level,
application.inperson_type,
application.location,
application.id,
)
return application

View File

@ -19,6 +19,9 @@ create table if not exists applications (
user_id text,
extra_data text,
job_level text default '',
-- remote - hyrbrid - in-person
inperson_type text default '',
location text default '',
status_id text default null,
agency boolean default false,
create_time timestamp default now ()

View File

@ -15,9 +15,13 @@ browser.runtime.onMessage.addListener((message) => {
Object.values(document.querySelector('button[class="job-details-preferences-and-skills"]')?.children ?? []).find(a => a.innerText.match(/\d/) && !a.innerText.match('skill'))?.innerText ??
'';
const inperson_type = (Object.values(document.querySelector('button[class="job-details-preferences-and-skills"]')?.children).find(a => a.innerText.match(/hybrid|remote|on-site/i))?.innerText?.split('\n') ?? [])[0]?.trim() ?? ''
const location = document.querySelector('div[class^="job-details-jobs-unified-top-card__primary-description-container"]')?.children[0]?.children[0]?.innerText ?? ''
const description = document.querySelector('article').textContent;
browser.runtime.sendMessage({ type: "GOT_INFO_R", company, jobTitle, money, description });
browser.runtime.sendMessage({ type: "GOT_INFO_R", company, jobTitle, money, description, location, inperson_type });
return;
}
// Ignore everything that is no glassdoor
@ -32,12 +36,14 @@ browser.runtime.onMessage.addListener((message) => {
let money = document.querySelectorAll('div[class^="SalaryEstimate_salaryRange"]')?.[0]?.innerText ?? '';
const location = document.querySelector('div[data-test="location"]')?.innerText ?? '';
const moneySectionNode = document.querySelector('section>section');
if (moneySectionNode && ["Base pay range", "Base pay"].includes(moneySectionNode.querySelector('h2')?.textContent)) {
money = moneySectionNode.querySelector("div>div>div").children[1]?.textContent ?? ''
}
browser.runtime.sendMessage({ type: "GOT_INFO_R", company, jobTitle, money, description });
browser.runtime.sendMessage({ type: "GOT_INFO_R", company, jobTitle, money, description, location, inperson_type: '' });
} else if (message.type === "GOT_INFO_R") {
window.postMessage(message);
}

View File

@ -34,6 +34,8 @@ export type Application = {
create_time: string;
status_history: string;
job_level: string;
location: string;
inperson_type: string;
flairs: Flair[];
events: ApplicationEvent[];
urls: string[];

View File

@ -15,10 +15,11 @@
let open = $state(false);
let ref: HTMLDivElement;
let btn: HTMLButtonElement;
function callOutside(e: MouseEvent) {
if (!ref) return;
if (!ref.contains(e.target as any)) return;
if (ref.contains(e.target as any) || btn.contains(e.target as any)) return;
open = false;
}
@ -31,7 +32,7 @@
</script>
<div class="relative">
<button class={buttonClass} onclick={() => (open = true)}>
<button class={buttonClass} onclick={() => (open = true)} bind:this={btn}>
{@render buttonChildren()}
</button>
{#if open}

View File

@ -328,8 +328,8 @@
<div class="bg-white p-2 text-black rounded-lg w-[190mm] print:text-sm">
<div>
<div>
<h1>BCompSc with First Class Honours @ University of Surrey</h1>
<div class="ml-5">
<h1>BSc Computer Science @ University of Surrey</h1>
<div class="ml-2 flex justify-between">
<h2>July 2020 - June 2024</h2>
</div>
</div>

View File

@ -212,17 +212,39 @@
{} as Record<string, number>
)}
/>
</div>
{#if flairStats !== 'loading'}
<div class="flex gap-5">
<Pie title={'Flair stats'} data={flairStats} sensitivity={0.02} />
<div class="max-h-[500px] overflow-auto">
<ul>
{#each Object.keys(flairStats).toSorted((a, b) => flairStats[b] - flairStats[a]) as flair}
<li>{flair}: {flairStats[flair]}</li>
{/each}
</ul>
</div>
<Pie
title={'Location'}
sensitivity={0.01}
data={applicationStore.all.reduce(
(acc, item) => {
const l = item.location === '' ? 'Unknown' : item.location;
if (acc[l]) {
acc[l] += 1;
} else {
acc[l] = 1;
}
return acc;
},
{} as Record<string, number>
)}
/>
<Pie
title={'In Person type'}
sensitivity={0.01}
data={applicationStore.all.reduce(
(acc, item) => {
const l =
item.inperson_type === '' ? 'Unknown' : item.inperson_type;
if (acc[l]) {
acc[l] += 1;
} else {
acc[l] = 1;
}
return acc;
},
{} as Record<string, number>
)}
/>
<Pie
title={'Job Level'}
data={applicationStore.all.reduce(
@ -239,6 +261,33 @@
)}
sensitivity={0.02}
/>
<Pie
title={'Agency'}
data={applicationStore.all.reduce(
(acc, a) => {
const i = a.agency ? 'Agency' : 'Direct';
if (acc[i]) {
acc[i] += 1;
} else {
acc[i] = 1;
}
return acc;
},
{} as Record<string, number>
)}
sensitivity={0.02}
/>
</div>
{#if flairStats !== 'loading'}
<div class="flex gap-5">
<Pie title={'Flair stats'} data={flairStats} sensitivity={0.02} />
<div class="max-h-[500px] overflow-auto">
<ul>
{#each Object.keys(flairStats).toSorted((a, b) => flairStats[b] - flairStats[a]) as flair}
<li>{flair}: {flairStats[flair]}</li>
{/each}
</ul>
</div>
</div>
{/if}
<div bind:clientWidth={width}>

View File

@ -45,9 +45,9 @@
dataf.push({
value: otherSum,
name: 'Other',
title: other
title: `Other: ${otherSum}\n${other
.toSorted((a, b) => b.value - a.value)
.reduce((acc, a) => `${acc}${a.name}: ${a.value}\n`, '')
.reduce((acc, a) => `${acc}${a.name}: ${a.value}\n`, '')}`
});
}

View File

@ -11,8 +11,11 @@
onreload: (item: Application) => void;
} = $props();
const JobLevels = ['intern', 'entry', 'junior', 'mid', 'senior', 'staff', 'lead', 'none'];
let filter = $state('');
let filterStatus: string[] = $state([]);
let jobLevels: string[] = $state([]);
let advFilters = $state(false);
type ExtraFilterType =
@ -62,6 +65,11 @@
return false;
}
const temp = jobLevels.map((a) => (a === 'none' ? '' : a));
if (temp.length !== 0 && !temp.includes(i.job_level ?? '')) {
return false;
}
if (!filter) {
return true;
}
@ -128,9 +136,10 @@
</div>
</div>
{#if advFilters}
<div>
<div class="flex gap-2">
<InplaceDialog
buttonClass="border-slate-300 border border-solid color-slate-300 p-1 rounded-md bg-slate-100/50"
dialogClass="w-[200px]"
>
{#snippet buttonChildren()}
<i class="bi bi-plus"></i> Status
@ -150,6 +159,26 @@
{/each}
</div>
</InplaceDialog>
<InplaceDialog
buttonClass="border-slate-300 border border-solid color-slate-300 p-1 rounded-md bg-slate-100/50"
dialogClass="w-[200px]"
>
{#snippet buttonChildren()}
<i class="bi bi-plus"></i> Job Level
{/snippet}
<div class="flex flex-wrap gap-2">
{#each JobLevels.filter((a) => !jobLevels.includes(a)) as jobLevel}
<button
class="border-red-300 border border-solid color-red-300 bg-red-100/50 p-1 rounded-md text-red-800"
onclick={() => {
jobLevels = [...jobLevels, jobLevel];
}}
>
{jobLevel}
</button>
{/each}
</div>
</InplaceDialog>
</div>
<h2>Filters</h2>
<div class="flex gap-2">
@ -163,6 +192,16 @@
{node.name}
</button>
{/each}
{#each jobLevels.filter((a) => jobLevels.includes(a)) as jobLevel}
<button
class="border-red-300 border border-solid color-red-300 bg-red-100/50 p-1 rounded-md text-red-800"
onclick={() => {
jobLevels = jobLevels.filter((a) => a != jobLevel);
}}
>
{jobLevel}
</button>
{/each}
{#each extraFiltersToDisplay as filter}
{#if filter.type === 'name'}
<span

View File

@ -136,6 +136,13 @@
applicationStore.all[activeItem].payrange = lastExtData.money;
}
applicationStore.all[activeItem].inperson_type = (
lastExtData.inperson_type as string
).toLowerCase();
applicationStore.all[activeItem].location = (lastExtData.location as string).split(',')[0];
console.log(lastExtData);
const title: string = lastExtData.jobTitle;
if (title.match(/intern|apprenticeship/i)) {
applicationStore.all[activeItem].job_level = 'intern';
@ -275,14 +282,45 @@
onchange={save}
/>
</fieldset>
<fieldset>
<label class="flabel" for="payrange">Pay Range</label>
<fieldset class="flex items-center gap-2">
<input
class="finput"
id="payrange"
bind:value={applicationStore.all[activeItem].payrange}
onchange={save}
/>
<button
title="Move hourly to yearly"
class="p-2"
onclick={() => {
const payrange = applicationStore.all[activeItem!].payrange
// remove decimal places
.replace(/(\d+).(\d+)/g, '$1')
.replace(/[kK]/g, '000')
.replace(/[^\d\-]/g, '')
.replace(//g, '-')
.split('-')
.map((a) => {
console.log('here', a);
return +a;
});
payrange[0] = payrange[0] * 8 * 5 * 4 * 12;
if (payrange[1] != undefined) {
payrange[1] = payrange[1] * 8 * 5 * 4 * 12;
applicationStore.all[activeItem!].payrange =
`${payrange[0]}-${payrange[1]}`;
window.requestAnimationFrame(() => save());
return;
}
applicationStore.all[activeItem!].payrange = `${payrange[0]}`;
window.requestAnimationFrame(() => save());
}}
>
<span class="bi bi-arrow-clockwise"></span>
</button>
</fieldset>
{#if !derivedItem.simple_url || showExtraData}
<fieldset class="max-w-full min-w-0 overflow-hidden">
@ -317,7 +355,8 @@
</div>
</div>
{/if}
<fieldset>
<div class="flex gap-2">
<fieldset class="flex-grow">
<label class="flabel" for="title">Job Level</label>
<select
class="finput"
@ -334,6 +373,29 @@
<option value="lead"> Lead </option>
</select>
</fieldset>
<fieldset class="flex-grow">
<label class="flabel" for="title">In Person?</label>
<select
class="finput"
id="inperson_type"
bind:value={applicationStore.all[activeItem].inperson_type}
onchange={save}
>
<option value="on-site"> On-site </option>
<option value="remote"> Remote </option>
<option value="hybrid"> Hybrid </option>
</select>
</fieldset>
<fieldset class="flex-grow">
<label class="flabel" for="title">Location</label>
<input
class="finput"
id="location"
bind:value={applicationStore.all[activeItem].location}
onchange={save}
/>
</fieldset>
</div>
<div>
<div class="flabel">Tags</div>
<div class="flex gap-2 flex-wrap">