About Us
-+ About Us +
- Meet the Team
-
-
-
- Meet the Team
-
-
-
+ Meet the Team
+
+
+
+ Meet the Team
+
+
+ Welcome to the Resource Center for Auditory Training!
-Looking for resource recommendations?
- - Search for Auditory Resources -+ Welcome to the Resource Center for Auditory Training! +
++ Looking for resource recommendations? +
+ + Search for Auditory Resources +You can use the
+ You can use the
Getting Started
++ Getting Started +
Our Purpose
++ Our Purpose +
The goal of this site is to provide resources for cochlear implant users to practice listening with their device. While cochlear implants are highly effective in providing access to speech sounds for patients, it can take time and practice for them to adjust to the new signal transmitted through the implant. Auditory training can assist cochlear implant users in practicing listening to environmental sounds, understanding speech sounds in both quiet and noisy environments, and (re)training to enjoy music. We have compiled and categorized all available auditory training resources for cochlear implant users, including smartphone, web-based, and other online resources that can be utilized as clinician-guided or home-based by the cochlear implant patient and their families. Our online index allows both professionals and patients to filter through the resources based on specific characteristics and requested features, making the selection process more efficient.
++ The goal of this site is to provide resources for cochlear + implant users to practice listening with their device. While + cochlear implants are highly effective in providing access to + speech sounds for patients, it can take time and practice for + them to adjust to the new signal transmitted through the + implant. Auditory training can assist cochlear implant users in + practicing listening to environmental sounds, understanding + speech sounds in both quiet and noisy environments, and + (re)training to enjoy music. We have compiled and categorized + all available auditory training resources for cochlear implant + users, including smartphone, web-based, and other online + resources that can be utilized as clinician-guided or home-based + by the cochlear implant patient and their families. Our online + index allows both professionals and patients to filter through + the resources based on specific characteristics and requested + features, making the selection process more efficient. +
Want to learn more?
-
+
+
+
+ Want to learn more?
+
+
+
+
Learn more about the project
-
-
+
+
About
-
+
-
-
+
+
Get in touch with the team
-
-
+
+
Contact
-
+
-
-
+
+
Tell us how we’re doing!
-
+
Give Feedback
-
+
-
+
>
);
};
diff --git a/src/pages/resources/[id].tsx b/src/pages/resources/[id].tsx
index 76d7773..1994a70 100644
--- a/src/pages/resources/[id].tsx
+++ b/src/pages/resources/[id].tsx
@@ -1,35 +1,35 @@
import { type InferGetStaticPropsType, type GetStaticPropsContext } from "next";
-import { GlobeAltIcon, DocumentIcon } from '@heroicons/react/24/solid';
-import { createProxySSGHelpers } from '@trpc/react-query/ssg';
+import { GlobeAltIcon, DocumentIcon } from "@heroicons/react/24/solid";
+import { createProxySSGHelpers } from "@trpc/react-query/ssg";
import { appRouter } from "~/server/api/root";
import { prisma } from "~/server/db";
import { api } from "~/utils/api";
import { ResourceDescription, ResourceInfo } from "~/components/ResourceTable";
import { type PlatformLink } from "@prisma/client";
-import Image from 'next/image';
+import Image from "next/image";
import Link from "next/link";
import Footer from "~/components/Footer";
import Header from "~/components/Header";
export const getStaticPaths = async () => {
- const resources = (await prisma.auditoryResource.findMany({
+ const resources = await prisma.auditoryResource.findMany({
select: {
id: true,
- }
- }));
+ },
+ });
return {
paths: resources.map((resource) => ({
params: {
id: resource.id,
- }
+ },
})),
- fallback: 'blocking',
- }
+ fallback: "blocking",
+ };
};
export async function getStaticProps(
- context: GetStaticPropsContext<{ id: string }>,
+ context: GetStaticPropsContext<{ id: string }>
) {
const ssg = createProxySSGHelpers({
router: appRouter,
@@ -40,7 +40,7 @@ export async function getStaticProps(
});
const id = context.params?.id as string;
- await ssg.auditoryResource.byId.prefetch({id});
+ await ssg.auditoryResource.byId.prefetch({ id });
return {
props: {
@@ -51,97 +51,120 @@ export async function getStaticProps(
};
}
-const PlatformLinkButton = ({platformLink}: {platformLink: PlatformLink}) => {
+const PlatformLinkButton = ({
+ platformLink,
+}: {
+ platformLink: PlatformLink;
+}) => {
const Inner = () => {
switch (platformLink.platform) {
case "APP_ANDROID": {
return (
-
- )
+
+ );
}
case "APP_IOS": {
return (
-
- )
+
+ );
}
case "PDF": {
return (
-
+
-
- Document
-
+ Document
- )
+ );
}
case "WEBSITE": {
return (
-
+
-
- Website
-
+ Website
- )
+ );
}
}
- }
+ };
return (
- )
-}
+ );
+};
-const DownloadButtons = ({platformLinks}: {platformLinks: PlatformLink[]}) => {
+const DownloadButtons = ({
+ platformLinks,
+}: {
+ platformLinks: PlatformLink[];
+}) => {
const buttons = platformLinks.map((paltformLink, index) => {
- return (
-
- )
+ return ;
});
-
- return (
-
- {buttons}
-
- )
-}
-const ResourceViewPage = (props: InferGetStaticPropsType) => {
+ return {buttons};
+};
+
+const ResourceViewPage = (
+ props: InferGetStaticPropsType
+) => {
const { id } = props;
const resourceQuery = api.auditoryResource.byId.useQuery({ id });
if (!resourceQuery.data) {
- return <>
- >
+ return <>>;
}
- return <>
-
-
-
-
-
-
- Links
-
+ return (
+ <>
+
+
+
+
+
+
+ Links
+
+
+
+
+
+
+
+
+
+
+ Ages {resourceQuery.data.ages.min}
+ {resourceQuery.data.ages.max >= 100
+ ? "+"
+ : `-${resourceQuery.data.ages.max}`}
+
+
-
-
-
-
-
-
- Ages {resourceQuery.data.ages.min}{resourceQuery.data.ages.max >= 100 ? "+" : `-${resourceQuery.data.ages.max}`}
-
-
-
-
-
-
- >
+
+
+
+ >
+ );
};
-export default ResourceViewPage
\ No newline at end of file
+export default ResourceViewPage;
diff --git a/src/pages/resources/index.tsx b/src/pages/resources/index.tsx
index d88cc8f..2d14a91 100644
--- a/src/pages/resources/index.tsx
+++ b/src/pages/resources/index.tsx
@@ -1,4 +1,4 @@
-import { LinkIcon } from '@heroicons/react/20/solid';
+import { LinkIcon } from "@heroicons/react/20/solid";
import Link from "next/link";
import { useRouter } from "next/router";
import ResourceTable from "~/components/ResourceTable";
@@ -8,22 +8,22 @@ import Footer from "~/components/Footer";
import Header from "~/components/Header";
const Resources = () => {
- const router = useRouter()
+ const router = useRouter();
const queryData = parseQueryData(router.query);
const currentPage = queryData.page;
const resourceQuery = api.auditoryResource.search.useQuery({
- skip: (queryData.page - 1) * queryData.perPage,
- take: queryData.perPage,
- ages: queryData.age,
- platforms: queryData.platforms,
- skill_levels: queryData.skill_levels,
- skills: queryData.skills,
+ skip: (queryData.page - 1) * queryData.perPage,
+ take: queryData.perPage,
+ ages: queryData.age,
+ platforms: queryData.platforms,
+ skill_levels: queryData.skill_levels,
+ skills: queryData.skills,
});
-
+
if (!resourceQuery.data) {
- return <>>
+ return <>>;
}
const totalPages = Math.ceil(resourceQuery.data.count / queryData.perPage);
@@ -31,24 +31,35 @@ const Resources = () => {
return (
<>
-
-
+
+
All Resources
Fill out the
-
+
search form
-
+
- for a list of auditory training resource recommendations.
+
+ {" "}
+ for a list of auditory training resource recommendations.
+
-
+
>
);
-}
+};
-export default Resources;
\ No newline at end of file
+export default Resources;
diff --git a/src/pages/resources/search.tsx b/src/pages/resources/search.tsx
index a85ae26..7b37de7 100644
--- a/src/pages/resources/search.tsx
+++ b/src/pages/resources/search.tsx
@@ -1,121 +1,127 @@
import Footer from "~/components/Footer";
import Header from "~/components/Header";
-import { GuidedSearch, type Question, type QuestionTypes } from "~/components/Search";
+import {
+ GuidedSearch,
+ type Question,
+ type QuestionTypes,
+} from "~/components/Search";
const questions: Question[] = [
- {
- for: "ages",
- header: "Age of Patient",
- question: "How old is the patient?",
- maxSelect: 1,
- optional: true,
- options: [
- {
- label: "Child (0-10)",
- value: "0-9",
- },
- {
- label: "Teen (10-20)",
- value: "10-20",
- },
- {
- label: "Adult (21+)",
- value: "21-100",
- },
- ],
- },
- {
- for: "platforms",
- header: "Desired Platforms",
- question: "What platform(s) does the resource need to be on?",
- optional: true,
- options: [
- {
- label: "Apple (iOS)",
- value: "APP_IOS",
- },
- {
- label: "Android",
- value: "APP_ANDROID",
- },
- {
- label: "Web-Based",
- value: "WEBSITE",
- },
- {
- label: "PDF (printable)",
- value: "PDF",
- }
- ]
- },
- {
- for: "skills",
- header: "Skills Practiced",
- question: "What skill(s) would you like the resource to cover?",
- optional: true,
- options: [
- {
- label: "Phonemes",
- value: "PHONEMES",
- },
- {
- label: "Words",
- value: "WORDS",
- },
- {
- label: "Sentence",
- value: "SENTENCES",
- },
- {
- label: "Discourse/Complex",
- value: "DISCOURSE",
- },
- {
- label: "Music",
- value: "MUSIC",
- },
- {
- label: "Environmental Sounds",
- value: "ENVIRONMENT",
- },
- ]
- },
- {
- for: "skill_levels",
- header: "Skill Level",
- question: "What skill level(s) should the resource have?",
- optional: true,
- options: [
- {
- label: "Beginner",
- value: "BEGINNER",
- },
- {
- label: "Intermediate",
- value: "INTERMEDIATE",
- },
- {
- label: "Advanced",
- value: "ADVANCED",
- }
- ]
- },
-]
+ {
+ for: "ages",
+ header: "Age of Patient",
+ question: "How old is the patient?",
+ maxSelect: 1,
+ optional: true,
+ options: [
+ {
+ label: "Child (0-10)",
+ value: "0-9",
+ },
+ {
+ label: "Teen (10-20)",
+ value: "10-20",
+ },
+ {
+ label: "Adult (21+)",
+ value: "21-100",
+ },
+ ],
+ },
+ {
+ for: "platforms",
+ header: "Desired Platforms",
+ question: "What platform(s) does the resource need to be on?",
+ optional: true,
+ options: [
+ {
+ label: "Apple (iOS)",
+ value: "APP_IOS",
+ },
+ {
+ label: "Android",
+ value: "APP_ANDROID",
+ },
+ {
+ label: "Web-Based",
+ value: "WEBSITE",
+ },
+ {
+ label: "PDF (printable)",
+ value: "PDF",
+ },
+ ],
+ },
+ {
+ for: "skills",
+ header: "Skills Practiced",
+ question: "What skill(s) would you like the resource to cover?",
+ optional: true,
+ options: [
+ {
+ label: "Phonemes",
+ value: "PHONEMES",
+ },
+ {
+ label: "Words",
+ value: "WORDS",
+ },
+ {
+ label: "Sentence",
+ value: "SENTENCES",
+ },
+ {
+ label: "Discourse/Complex",
+ value: "DISCOURSE",
+ },
+ {
+ label: "Music",
+ value: "MUSIC",
+ },
+ {
+ label: "Environmental Sounds",
+ value: "ENVIRONMENT",
+ },
+ ],
+ },
+ {
+ for: "skill_levels",
+ header: "Skill Level",
+ question: "What skill level(s) should the resource have?",
+ optional: true,
+ options: [
+ {
+ label: "Beginner",
+ value: "BEGINNER",
+ },
+ {
+ label: "Intermediate",
+ value: "INTERMEDIATE",
+ },
+ {
+ label: "Advanced",
+ value: "ADVANCED",
+ },
+ ],
+ },
+];
const SearchPage = () => {
- return <>
-
-
-
-
-
-
-
-
-
-
+ return (
+ <>
+
+
+
+
+
+
+
+
+
+
>
-}
+ );
+};
-export default SearchPage
\ No newline at end of file
+export default SearchPage;
diff --git a/src/server/api/routers/auditoryResources.ts b/src/server/api/routers/auditoryResources.ts
index 46dede4..bbd6d62 100644
--- a/src/server/api/routers/auditoryResources.ts
+++ b/src/server/api/routers/auditoryResources.ts
@@ -1,9 +1,12 @@
-import { SkillLevel, Skill, Platform, type AuditoryResource } from "@prisma/client";
+import {
+ SkillLevel,
+ Skill,
+ Platform,
+ type AuditoryResource,
+} from "@prisma/client";
import { z } from "zod";
-import {
- createTRPCRouter, publicProcedure,
-} from "~/server/api/trpc";
+import { createTRPCRouter, publicProcedure } from "~/server/api/trpc";
export const auditoryResourceRouter = createTRPCRouter({
byId: publicProcedure
@@ -12,29 +15,33 @@ export const auditoryResourceRouter = createTRPCRouter({
const resource = await ctx.prisma.auditoryResource.findUnique({
where: {
id: input.id,
- }
+ },
});
-
+
return { ...resource } as AuditoryResource;
}),
getAll: publicProcedure.query(({ ctx }) => {
- return ctx.prisma.auditoryResource.findMany();
+ return ctx.prisma.auditoryResource.findMany();
}),
search: publicProcedure
- .input(z.object({
- take: z.number().int(),
- skip: z.number().int(),
- ages: z.object({
- min: z.number().int(),
- max: z.number().int(),
- }).optional(),
- platforms: z.nativeEnum(Platform).array().optional(),
- skill_levels: z.nativeEnum(SkillLevel).array().optional(),
- skills: z.nativeEnum(Skill).array().optional(),
- }))
- .query(async ({ input, ctx}) => {
+ .input(
+ z.object({
+ take: z.number().int(),
+ skip: z.number().int(),
+ ages: z
+ .object({
+ min: z.number().int(),
+ max: z.number().int(),
+ })
+ .optional(),
+ platforms: z.nativeEnum(Platform).array().optional(),
+ skill_levels: z.nativeEnum(SkillLevel).array().optional(),
+ skills: z.nativeEnum(Skill).array().optional(),
+ })
+ )
+ .query(async ({ input, ctx }) => {
const search = {
ages: {
is: {
@@ -43,8 +50,8 @@ export const auditoryResourceRouter = createTRPCRouter({
},
max: {
gte: input.ages?.max,
- }
- }
+ },
+ },
},
skill_levels: {
hasEvery: input.skill_levels ?? [],
@@ -56,10 +63,10 @@ export const auditoryResourceRouter = createTRPCRouter({
some: {
platform: {
in: input.platforms,
- }
- }
- }
- }
+ },
+ },
+ },
+ };
const [count, resources] = await ctx.prisma.$transaction([
ctx.prisma.auditoryResource.count({
@@ -69,12 +76,12 @@ export const auditoryResourceRouter = createTRPCRouter({
skip: input.skip,
take: input.take,
where: search,
- })
+ }),
]);
return {
count,
resources,
- }
+ };
}),
});
diff --git a/src/utils/enumWordLut.ts b/src/utils/enumWordLut.ts
index d6bf76c..087861f 100644
--- a/src/utils/enumWordLut.ts
+++ b/src/utils/enumWordLut.ts
@@ -4,48 +4,48 @@ import { type Skill, type Platform } from "@prisma/client";
* Takes a platform enum and translates it to readable form.
*/
export const translateEnumPlatform = (value: Platform) => {
- switch(value) {
- case "APP_ANDROID": {
- return "Android";
- }
- case "APP_IOS": {
- return "Apple";
- }
- case "PDF": {
- return "PDF Document";
- }
- case "WEBSITE": {
- return "Website"
- }
+ switch (value) {
+ case "APP_ANDROID": {
+ return "Android";
}
-}
+ case "APP_IOS": {
+ return "Apple";
+ }
+ case "PDF": {
+ return "PDF Document";
+ }
+ case "WEBSITE": {
+ return "Website";
+ }
+ }
+};
/**
* Takes a skill enum value and translates it to human text
- * @param value
+ * @param value
*/
export const translateEnumSkill = (value: Skill) => {
- switch(value) {
- case "ENVIRONMENT": {
- return "Environmental Sounds";
- }
- case "BACKGROUND": {
- return "Background";
- }
- case "DISCOURSE": {
- return "Discourse/Complex";
- }
- case "MUSIC": {
- return "Music Appreciation"
- }
- case "PHONEMES": {
- return "Phonemes";
- }
- case "SENTENCES": {
- return "Sentences"
- }
- case "WORDS": {
- return "Words"
- }
+ switch (value) {
+ case "ENVIRONMENT": {
+ return "Environmental Sounds";
}
-}
\ No newline at end of file
+ case "BACKGROUND": {
+ return "Background";
+ }
+ case "DISCOURSE": {
+ return "Discourse/Complex";
+ }
+ case "MUSIC": {
+ return "Music Appreciation";
+ }
+ case "PHONEMES": {
+ return "Phonemes";
+ }
+ case "SENTENCES": {
+ return "Sentences";
+ }
+ case "WORDS": {
+ return "Words";
+ }
+ }
+};
diff --git a/src/utils/parseSearchForm.ts b/src/utils/parseSearchForm.ts
index eed0b7b..c2034b9 100644
--- a/src/utils/parseSearchForm.ts
+++ b/src/utils/parseSearchForm.ts
@@ -1,75 +1,80 @@
-import { type Platform, type RangeInput, type Skill, type SkillLevel } from "@prisma/client";
+import {
+ type Platform,
+ type RangeInput,
+ type Skill,
+ type SkillLevel,
+} from "@prisma/client";
import { type ParsedUrlQuery } from "querystring";
export interface ViewDetails {
- page: number;
- perPage: number;
+ page: number;
+ perPage: number;
}
-
+
export interface SearchQuery {
- age?: RangeInput,
- platforms?: Platform[],
- skill_levels?: SkillLevel[],
- skills?: Skill[],
+ age?: RangeInput;
+ platforms?: Platform[];
+ skill_levels?: SkillLevel[];
+ skills?: Skill[];
}
export type ParsedQueryData = SearchQuery & ViewDetails;
export const parseQueryData = (query: ParsedUrlQuery): ParsedQueryData => {
- const view = {
- page: Number(query["page"] ?? 1),
- perPage: Number(query["perPage"] ?? 10),
- }
- const filter: SearchQuery = {};
+ const view = {
+ page: Number(query["page"] ?? 1),
+ perPage: Number(query["perPage"] ?? 10),
+ };
+ const filter: SearchQuery = {};
- if (query["ages"]) {
- const ages: number[] = [];
+ if (query["ages"]) {
+ const ages: number[] = [];
- if (Array.isArray(query["ages"])) {
- const validRanges = query["ages"].filter((value) => {
- return value.split("-").length == 2;
- });
+ if (Array.isArray(query["ages"])) {
+ const validRanges = query["ages"].filter((value) => {
+ return value.split("-").length == 2;
+ });
- validRanges.forEach((value) => {
- const split = value.split("-");
- ages.push(Number(split[0]));
- ages.push(Number(split[1]));
- });
- } else {
- const split = query["ages"].split("-");
+ validRanges.forEach((value) => {
+ const split = value.split("-");
ages.push(Number(split[0]));
ages.push(Number(split[1]));
- }
-
- filter.age = {
- min: Math.min(...ages),
- max: Math.max(...ages),
- }
+ });
+ } else {
+ const split = query["ages"].split("-");
+ ages.push(Number(split[0]));
+ ages.push(Number(split[1]));
}
- if (query["platforms"]) {
- if (Array.isArray(query["platforms"])) {
- filter.platforms = query["platforms"] as Platform[];
- } else {
- filter.platforms = [query["platforms"]] as Platform[];
- }
- }
+ filter.age = {
+ min: Math.min(...ages),
+ max: Math.max(...ages),
+ };
+ }
- if (query["skill_levels"]) {
- if (Array.isArray(query["skill_levels"])) {
- filter.skill_levels = query["skill_levels"] as SkillLevel[];
- } else {
- filter.skill_levels = [query["skill_levels"]] as SkillLevel[];
- }
+ if (query["platforms"]) {
+ if (Array.isArray(query["platforms"])) {
+ filter.platforms = query["platforms"] as Platform[];
+ } else {
+ filter.platforms = [query["platforms"]] as Platform[];
}
+ }
- if (query["skills"]) {
- if (Array.isArray(query["skills"])) {
- filter.skills = query["skills"] as Skill[];
- } else {
- filter.skills = [query["skills"]] as Skill[];
- }
+ if (query["skill_levels"]) {
+ if (Array.isArray(query["skill_levels"])) {
+ filter.skill_levels = query["skill_levels"] as SkillLevel[];
+ } else {
+ filter.skill_levels = [query["skill_levels"]] as SkillLevel[];
}
+ }
- return {...filter, ...view};
-}
\ No newline at end of file
+ if (query["skills"]) {
+ if (Array.isArray(query["skills"])) {
+ filter.skills = query["skills"] as Skill[];
+ } else {
+ filter.skills = [query["skills"]] as Skill[];
+ }
+ }
+
+ return { ...filter, ...view };
+};
+ Want to learn more? +
+Learn more about the project
- - + + About -
+
+
Get in touch with the team
-
-
+
+
Contact
-
+
-
-
+
+
Tell us how we’re doing!
-
+
Give Feedback
-
+
Get in touch with the team
- - + + Contact -
+
+
Tell us how we’re doing!
-
+
Give Feedback
-
+
Tell us how we’re doing!
- + Give Feedback -Links
-Links
+All Resources
Fill out the
- + search form -for a list of auditory training resource recommendations.
++ {" "} + for a list of auditory training resource recommendations. +