feat: Work in progress charts endpoint
This commit is contained in:
parent
2ee82cdd15
commit
a69284814c
1001
package-lock.json
generated
1001
package-lock.json
generated
File diff suppressed because it is too large
Load Diff
|
@ -81,7 +81,8 @@
|
|||
"tailwindcss": "^3.0.22",
|
||||
"textversionjs": "^1.1.3",
|
||||
"ts-node": "^10.7.0",
|
||||
"tunnel": "^0.0.6"
|
||||
"tunnel": "^0.0.6",
|
||||
"victory": "^36.3.2"
|
||||
},
|
||||
"devDependencies": {
|
||||
"@netlify/plugin-nextjs": "^4.2.4",
|
||||
|
|
71
src/pages/charts/embed/[id].tsx
Normal file
71
src/pages/charts/embed/[id].tsx
Normal file
|
@ -0,0 +1,71 @@
|
|||
import { GetServerSideProps, NextPage } from "next";
|
||||
import Error from "next/error";
|
||||
|
||||
import { DashboardItem } from "../../../backend/dashboards";
|
||||
import { DisplayForecasts } from "../../../web/display/DisplayForecasts";
|
||||
import { FrontendForecast } from "../../../web/platforms";
|
||||
import { getDashboardForecastsByDashboardId } from "../../../web/worker/getDashboardForecasts";
|
||||
import { reqToBasePath } from "../../../web/utils";
|
||||
|
||||
interface Props {
|
||||
dashboardForecasts: FrontendForecast[];
|
||||
dashboardItem: DashboardItem;
|
||||
numCols?: number;
|
||||
}
|
||||
|
||||
export const getServerSideProps: GetServerSideProps<Props> = async (
|
||||
context
|
||||
) => {
|
||||
const dashboardId = context.query.id as string;
|
||||
const numCols = Number(context.query.numCols);
|
||||
|
||||
const { dashboardItem, dashboardForecasts } =
|
||||
await getDashboardForecastsByDashboardId({
|
||||
dashboardId,
|
||||
basePath: reqToBasePath(context.req), // required on server side to find the API endpoint
|
||||
});
|
||||
|
||||
if (!dashboardItem) {
|
||||
context.res.statusCode = 404;
|
||||
}
|
||||
|
||||
return {
|
||||
props: {
|
||||
dashboardForecasts,
|
||||
dashboardItem,
|
||||
numCols: !numCols ? null : numCols < 5 ? numCols : 4,
|
||||
},
|
||||
};
|
||||
};
|
||||
|
||||
const EmbedDashboardPage: NextPage<Props> = ({
|
||||
dashboardForecasts,
|
||||
dashboardItem,
|
||||
numCols,
|
||||
}) => {
|
||||
if (!dashboardItem) {
|
||||
return <Error statusCode={404} />;
|
||||
}
|
||||
|
||||
return (
|
||||
<div className="mb-4 mt-3 flex flex-row justify-left items-center">
|
||||
<div className="mx-2 place-self-left">
|
||||
<div
|
||||
className={`grid grid-cols-${numCols || 1} sm:grid-cols-${
|
||||
numCols || 1
|
||||
} md:grid-cols-${numCols || 2} lg:grid-cols-${
|
||||
numCols || 3
|
||||
} gap-4 mb-6`}
|
||||
>
|
||||
<DisplayForecasts
|
||||
results={dashboardForecasts}
|
||||
numDisplay={dashboardForecasts.length}
|
||||
showIdToggle={false}
|
||||
/>
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
);
|
||||
};
|
||||
|
||||
export default EmbedDashboardPage;
|
51
src/pages/charts/index.tsx
Normal file
51
src/pages/charts/index.tsx
Normal file
|
@ -0,0 +1,51 @@
|
|||
import axios from "axios";
|
||||
import { NextPage } from "next";
|
||||
import { useRouter } from "next/router";
|
||||
|
||||
import { DashboardCreator } from "../../web/display/DashboardCreator";
|
||||
import { InfoBox } from "../../web/display/InfoBox";
|
||||
|
||||
import { Layout } from "../../web/display/Layout";
|
||||
import { LineHeader } from "../../web/display/LineHeader";
|
||||
|
||||
const DashboardsPage: NextPage = () => {
|
||||
const router = useRouter();
|
||||
|
||||
const handleSubmit = async (data) => {
|
||||
// Send to server to create
|
||||
// Get back the id
|
||||
let response = await axios({
|
||||
url: "/api/create-dashboard-from-ids",
|
||||
method: "POST",
|
||||
headers: { "Content-Type": "application/json" },
|
||||
data: JSON.stringify(data),
|
||||
}).then((res) => res.data);
|
||||
await router.push(`/dashboards/view/${response.dashboardId}`);
|
||||
};
|
||||
|
||||
return (
|
||||
<Layout page="dashboard">
|
||||
<div className="flex flex-col my-8 space-y-8">
|
||||
<LineHeader>Charts!</LineHeader>
|
||||
|
||||
<div className="self-center">{""}</div>
|
||||
<div className="max-w-2xl self-center">
|
||||
<InfoBox>
|
||||
This is an under-construction endpoint to display charts. Go to
|
||||
metaforecast.org/charts/view/[id] to find the chart for a particular
|
||||
question.
|
||||
</InfoBox>
|
||||
</div>
|
||||
<div className="max-w-2xl self-center">
|
||||
<InfoBox>
|
||||
You can find the necessary ids by toggling the advanced options in
|
||||
the search, or by visiting{" "}
|
||||
<a href="/api/all-forecasts">/api/all-forecasts</a>
|
||||
</InfoBox>
|
||||
</div>
|
||||
</div>
|
||||
</Layout>
|
||||
);
|
||||
};
|
||||
|
||||
export default DashboardsPage;
|
55
src/pages/charts/view/[id].tsx
Normal file
55
src/pages/charts/view/[id].tsx
Normal file
|
@ -0,0 +1,55 @@
|
|||
/* Imports */
|
||||
|
||||
import { GetServerSideProps, NextPage } from "next";
|
||||
import { Layout } from "../../../web/display/Layout";
|
||||
|
||||
import React from "react";
|
||||
|
||||
import { platforms } from "../../../backend/platforms";
|
||||
import { HistoryChart } from "../../../web/display/HistoryChart";
|
||||
import { FrontendForecast } from "../../../web/platforms";
|
||||
import searchAccordingToQueryData from "../../../web/worker/searchAccordingToQueryData";
|
||||
|
||||
interface Props {
|
||||
question: FrontendForecast;
|
||||
history: number[];
|
||||
}
|
||||
|
||||
export const getServerSideProps: GetServerSideProps<Props> = async (
|
||||
context
|
||||
) => {
|
||||
let urlQuery = context.query; // this is an object, not a string which I have to parse!!
|
||||
|
||||
let initialQueryParameters = {
|
||||
query: "test",
|
||||
starsThreshold: 2,
|
||||
forecastsThreshold: 0,
|
||||
forecastingPlatforms: platforms.map((platform) => platform.name),
|
||||
...urlQuery,
|
||||
};
|
||||
|
||||
let results: FrontendForecast[] = [];
|
||||
if (initialQueryParameters.query != "") {
|
||||
results = await searchAccordingToQueryData(initialQueryParameters, 1);
|
||||
console.log(results);
|
||||
}
|
||||
|
||||
return {
|
||||
props: {
|
||||
question: results[0] || null,
|
||||
history: [1, 2, 3, 4, 5, 6, 7, 8, 9, 10],
|
||||
},
|
||||
};
|
||||
};
|
||||
|
||||
const Chart: NextPage<Props> = ({ question, history }) => {
|
||||
return (
|
||||
<Layout page={"chart"}>
|
||||
<div className="w-6/12 mb-4 mt-8 flex flex-row justify-center items-center bg-white">
|
||||
<HistoryChart question={question} history={history} />
|
||||
</div>
|
||||
</Layout>
|
||||
);
|
||||
};
|
||||
|
||||
export default Chart;
|
132
src/web/display/HistoryChart.tsx
Normal file
132
src/web/display/HistoryChart.tsx
Normal file
|
@ -0,0 +1,132 @@
|
|||
import React from "react";
|
||||
|
||||
import { FrontendForecast } from "../platforms";
|
||||
import * as V from "victory";
|
||||
import {
|
||||
VictoryBar,
|
||||
VictoryLabel,
|
||||
VictoryTooltip,
|
||||
VictoryLine,
|
||||
VictoryScatter,
|
||||
VictoryChart,
|
||||
VictoryTheme,
|
||||
VictoryAxis,
|
||||
VictoryGroup,
|
||||
VictoryVoronoiContainer,
|
||||
} from "victory";
|
||||
|
||||
interface Props {
|
||||
question: FrontendForecast;
|
||||
history: number[];
|
||||
}
|
||||
|
||||
const data = [
|
||||
{ date: 1, probability: 0.1 },
|
||||
{ date: 2, probability: 0.2 },
|
||||
{ date: 3, probability: 0.4 },
|
||||
{ date: 4, probability: 0.6 },
|
||||
{ date: 5, probability: 0.6 },
|
||||
{ date: 6, probability: 0.65 },
|
||||
{ date: 7, probability: 0.65 },
|
||||
{ date: 8, probability: 0.65 },
|
||||
{ date: 9, probability: 0.7 },
|
||||
];
|
||||
|
||||
let getDate = (x) => {
|
||||
let date = new Date(x);
|
||||
return date.toISOString().slice(5, 10).replaceAll("-", "/");
|
||||
};
|
||||
|
||||
let dataAsXy = data.map((datum) => ({
|
||||
x: getDate(datum.date * (1000 * 60 * 60 * 24)),
|
||||
y: datum.probability,
|
||||
}));
|
||||
|
||||
export const HistoryChart: React.FC<Props> = ({ question, history }) => {
|
||||
return (
|
||||
<VictoryChart
|
||||
domainPadding={20}
|
||||
theme={VictoryTheme.material}
|
||||
height={300}
|
||||
containerComponent={<VictoryVoronoiContainer />}
|
||||
domain={{
|
||||
y: [0, 1],
|
||||
}}
|
||||
>
|
||||
<VictoryGroup
|
||||
color="#c43a31"
|
||||
data={dataAsXy}
|
||||
labels={({ datum }) => `${datum.y * 100}%`}
|
||||
labelComponent={
|
||||
<VictoryTooltip style={{ fontSize: 10, fill: "black" }} />
|
||||
}
|
||||
>
|
||||
<VictoryLine />
|
||||
<VictoryScatter size={({ active }) => (active ? 8 : 3)} />
|
||||
</VictoryGroup>
|
||||
<VictoryAxis
|
||||
// tickValues specifies both the number of ticks and where
|
||||
// they are placed on the axis
|
||||
tickValues={data.map((datum) => datum.date)}
|
||||
tickFormat={dataAsXy.map((datum) => datum.x)}
|
||||
style={{
|
||||
grid: { stroke: null, strokeWidth: 0.5 },
|
||||
}}
|
||||
tickLabelComponent={
|
||||
<VictoryLabel
|
||||
dy={0}
|
||||
angle={-30}
|
||||
style={{ fontSize: 7, fill: "gray" }}
|
||||
/>
|
||||
}
|
||||
/>
|
||||
<VictoryAxis
|
||||
dependentAxis
|
||||
// tickFormat specifies how ticks should be displayed
|
||||
tickFormat={(x) => `${x * 100}%`}
|
||||
style={{
|
||||
grid: { stroke: "#D3D3D3", strokeWidth: 0.5 },
|
||||
}}
|
||||
/>
|
||||
</VictoryChart>
|
||||
);
|
||||
};
|
||||
|
||||
/*
|
||||
<VictoryChart
|
||||
// domainPadding will add space to each side of VictoryBar to
|
||||
// prevent it from overlapping the axis
|
||||
domainPadding={20}
|
||||
theme={VictoryTheme.material}
|
||||
height={300}
|
||||
title={"Blah"}
|
||||
containerComponent={<VictoryVoronoiContainer />}
|
||||
>
|
||||
<VictoryGroup
|
||||
data={data.map((datum) => ({ x: datum.date, y: datum.probability }))}
|
||||
labels={data.map((datum) => `1%`)}
|
||||
style={{ labels: { fill: "black", fontSize: 10 } }}
|
||||
labelComponent={
|
||||
<VictoryTooltip style={{ fontSize: 10, fill: "black" }} dy={-15} />
|
||||
}
|
||||
>
|
||||
<VictoryLine
|
||||
height={300}
|
||||
width={300}
|
||||
style={{
|
||||
data: { stroke: "#000080" },
|
||||
parent: { border: "1px solid #ccc" },
|
||||
}}
|
||||
domain={{
|
||||
y: [0, 1],
|
||||
}}
|
||||
></VictoryLine>
|
||||
<VictoryScatter
|
||||
style={{
|
||||
data: { fill: "#000080" },
|
||||
}}
|
||||
size={3}
|
||||
/>
|
||||
</VictoryGroup>
|
||||
</VictoryChart>
|
||||
*/
|
Loading…
Reference in New Issue
Block a user