diff --git a/.htaccess b/.htaccess
index 9fded1e..4a897c9 100644
--- a/.htaccess
+++ b/.htaccess
@@ -1 +1,5 @@
-ErrorDocument 404 /404.html
\ No newline at end of file
+ErrorDocument 404 /404.html
+
+
+ Header add Access-Control-Allow-Origin 'https://api.martials.no/'
+
diff --git a/package-lock.json b/package-lock.json
index 6190cd0..7c115ba 100644
--- a/package-lock.json
+++ b/package-lock.json
@@ -1,23 +1,25 @@
{
- "name": "vite-template-solid",
+ "name": "martials-no",
"version": "0.1",
"lockfileVersion": 3,
"requires": true,
"packages": {
"": {
- "name": "vite-template-solid",
+ "name": "martials-no",
"version": "0.1",
"license": "MIT",
"dependencies": {
- "solid-js": "^1.6.2"
+ "solid-headless": "^0.13.0",
+ "solid-heroicons": "^3.1.1",
+ "solid-js": "^1.6.6"
},
"devDependencies": {
"autoprefixer": "^10.4.13",
- "postcss": "^8.4.19",
+ "postcss": "^8.4.20",
"tailwindcss": "^3.2.4",
- "typescript": "^4.9.3",
- "vite": "^3.2.4",
- "vite-plugin-solid": "^2.4.0"
+ "typescript": "^4.9.4",
+ "vite": "^3.2.5",
+ "vite-plugin-solid": "^2.5.0"
}
},
"node_modules/@ampproject/remapping": {
@@ -1965,6 +1967,31 @@
"semver": "bin/semver.js"
}
},
+ "node_modules/solid-headless": {
+ "version": "0.13.0",
+ "resolved": "https://registry.npmjs.org/solid-headless/-/solid-headless-0.13.0.tgz",
+ "integrity": "sha512-N118iPl7W5lyug4K+4c0cwM8YR2yEVJ/3H4wq6tVijATHfXfLnB1ACFI5GGHSXlwu70PUzaGSKwcVqLlloUMyA==",
+ "dependencies": {
+ "solid-use": "^0.5.0"
+ },
+ "engines": {
+ "node": ">=10"
+ },
+ "peerDependencies": {
+ "solid-js": "^1.2"
+ }
+ },
+ "node_modules/solid-heroicons": {
+ "version": "3.1.1",
+ "resolved": "https://registry.npmjs.org/solid-heroicons/-/solid-heroicons-3.1.1.tgz",
+ "integrity": "sha512-wfU/SqxqxWxInvfFlKJfCBPnJ94Zq1GQFiFL3M0KwZzT81lSF5yIC4HA/Czp3DYrn+dUumjO0qdrx8Ab3cy2sA==",
+ "dependencies": {
+ "solid-js": "^1.6.2"
+ },
+ "peerDependencies": {
+ "solid-js": ">= ^1.2.5"
+ }
+ },
"node_modules/solid-js": {
"version": "1.6.6",
"resolved": "https://registry.npmjs.org/solid-js/-/solid-js-1.6.6.tgz",
@@ -1987,6 +2014,17 @@
"solid-js": "^1.3"
}
},
+ "node_modules/solid-use": {
+ "version": "0.5.0",
+ "resolved": "https://registry.npmjs.org/solid-use/-/solid-use-0.5.0.tgz",
+ "integrity": "sha512-z1AaS99SL3gNP70K4+XIVKGXDRI56m3Q6Q4X63bfgw0kaN6cwT4SAyFZAmIRjHcvf5rOMRQfNgUV4YVDha+LcQ==",
+ "engines": {
+ "node": ">=10"
+ },
+ "peerDependencies": {
+ "solid-js": "^1.2"
+ }
+ },
"node_modules/source-map-js": {
"version": "1.0.2",
"resolved": "https://registry.npmjs.org/source-map-js/-/source-map-js-1.0.2.tgz",
diff --git a/package.json b/package.json
index 0c91e72..e026cb8 100644
--- a/package.json
+++ b/package.json
@@ -17,6 +17,8 @@
"vite-plugin-solid": "^2.5.0"
},
"dependencies": {
+ "solid-headless": "^0.13.0",
+ "solid-heroicons": "^3.1.1",
"solid-js": "^1.6.6"
}
}
diff --git a/simplify-truths.html b/simplify-truths.html
new file mode 100644
index 0000000..fefba88
--- /dev/null
+++ b/simplify-truths.html
@@ -0,0 +1,18 @@
+
+
+
+
+
+
+ Simplify | Martials.no
+
+
+
+
+
+
+
+
+
+
+
diff --git a/src/components/input.tsx b/src/components/input.tsx
new file mode 100644
index 0000000..3fdb1e5
--- /dev/null
+++ b/src/components/input.tsx
@@ -0,0 +1,113 @@
+/* @refresh reload */
+import { type Component, createSignal, JSX, Setter } from "solid-js";
+import type { InputProps } from "../types/interfaces";
+import Row from "./row";
+
+function setupEventListener(id: string, setIsHover: Setter): () => void {
+ let isMounted = true;
+
+ function hover(hover: boolean): void {
+ if (isMounted) {
+ setIsHover(hover);
+ }
+ }
+
+ const el = document.getElementById(id);
+ el?.addEventListener("pointerenter", () => hover(true));
+ el?.addEventListener("pointerleave", () => hover(false));
+
+ return () => {
+ el?.removeEventListener("pointerenter", () => hover(true));
+ el?.removeEventListener("pointerleave", () => hover(false));
+ isMounted = false;
+ }
+}
+
+/**
+ * Sets isText to 'true' or 'false' using the setIsText function.
+ * if the value of the input element is not empty and it's different from the current value
+ */
+function setSetIsText(id: string | undefined, isText: boolean, setIsText: Setter): void {
+ if (id) {
+ const el = document.getElementById(id) as HTMLInputElement | HTMLTextAreaElement | null;
+ if (el && el.value !== "" !== isText) {
+ setIsText(el.value !== "");
+ }
+ }
+}
+
+interface Input extends InputProps {
+ leading?: JSX.Element,
+ trailing?: JSX.Element,
+}
+
+export const Input: Component> = (
+ {
+ className,
+ id,
+ name,
+ type = "text",
+ title,
+ placeholder,
+ required = false,
+ onChange,
+ leading,
+ trailing
+ }): JSX.Element => {
+
+ /**
+ * Is 'true' if the input element is in focus
+ */
+ const [isFocused, setIsFocused] = createSignal(false);
+ /**
+ * Is 'true' if the user is hovering over the input element
+ */
+ const [isHover, setIsHover] = createSignal(false);
+ /**
+ * Is 'true' if the input element contains any characters
+ */
+ const [isText, setIsText] = createSignal(false);
+
+ document.addEventListener("DOMContentLoaded", () => {
+ if (id && title) {
+ setupEventListener(id, setIsHover);
+ }
+ });
+
+ return (
+
+ { leading }
+
+ setIsFocused(true) }
+ onBlur={ () => setIsFocused(false) }
+ name={ name ?? undefined }
+ type={ type }
+ placeholder={ placeholder ?? undefined }
+ required={ required }
+ onInput={ () => setSetIsText(id, isText(), setIsText) }
+ onChange={ onChange /*TODO only called after ENTER*/ }/>
+ { trailing }
+
+ );
+}
+
+function HoverTitle(
+ {
+ title,
+ isActive = false,
+ htmlFor
+ }: { title?: string | null, isActive?: boolean, htmlFor?: string }): JSX.Element {
+ return (
+
+ );
+}
diff --git a/src/components/output.tsx b/src/components/output.tsx
new file mode 100644
index 0000000..1d77adb
--- /dev/null
+++ b/src/components/output.tsx
@@ -0,0 +1,75 @@
+import { Disclosure, DisclosureButton, DisclosurePanel, Transition } from "solid-headless";
+import { Icon } from "solid-heroicons";
+import type { ChildProps, TitleProps } from "../types/interfaces";
+import { Component, JSX } from "solid-js";
+import { chevronUp } from "solid-heroicons/solid";
+
+interface InfoBoxProps extends TitleProps {
+ error?: boolean,
+}
+
+export const InfoBox: Component = (
+ {
+ title = "",
+ children,
+ error = false,
+ className
+ }: InfoBoxProps): JSX.Element => {
+ return (
+
+
{ title }
+
{ children }
+
+ );
+}
+
+interface MyDisclosureProps extends TitleProps {
+ defaultOpen?: boolean,
+ onClick?: JSX.EventHandlerUnion,
+}
+
+export const MyDisclosure: Component = (
+ {
+ title,
+ children,
+ defaultOpen = false,
+ className,
+ id,
+ onClick
+ }): JSX.Element => {
+ return (
+
+
+ { ({ isOpen }) =>
+ <>
+
+ { title }
+
+
+
+
+ { children }
+
+
+ >
+ }
+
+
+ );
+}
+
+export const MyDisclosureContainer: Component = ({ children, className }): JSX.Element => {
+ return (
+
+ { children }
+
+ );
+}
diff --git a/src/components/row.tsx b/src/components/row.tsx
new file mode 100644
index 0000000..d997c0b
--- /dev/null
+++ b/src/components/row.tsx
@@ -0,0 +1,9 @@
+/* @refresh reload */
+import { type Component } from "solid-js";
+import type { ChildProps } from "../types/interfaces";
+
+const Row: Component = ({ children, className }) => {
+ return { children }
+}
+
+export default Row;
diff --git a/src/components/truth-table.tsx b/src/components/truth-table.tsx
new file mode 100644
index 0000000..ba49f01
--- /dev/null
+++ b/src/components/truth-table.tsx
@@ -0,0 +1,57 @@
+/* @refresh reload */
+import type { SimpleProps } from "../types/interfaces";
+import type { Table } from "../types/interfaces";
+import { For } from "solid-js/web";
+import { type Component } from "solid-js";
+
+interface TruthTableProps extends SimpleProps {
+ table?: Table,
+ header?: string[],
+}
+
+const TruthTable: Component = (
+ {
+ table,
+ header,
+ className,
+ style,
+ id,
+ }) => {
+
+ return (
+
+
+
+
+ { (exp) => (
+
+ { exp }
+ |
+ ) }
+
+
+
+
+
+ { (row) =>
+
+
+ { (value) =>
+
+ { value ? "T" : "F" }
+ |
+ }
+
+
+ }
+
+
+
+ );
+}
+
+export default TruthTable;
diff --git a/src/index.css b/src/index.css
index dc91577..d7464a3 100644
--- a/src/index.css
+++ b/src/index.css
@@ -9,6 +9,18 @@
@apply after:content-['DEBUG'] after:absolute;
}
+ .flex-row-center {
+ @apply flex flex-row items-center;
+ }
+
+ .border-rounded {
+ @apply border rounded-2xl border-gray-700;
+ }
+
+ input[type="submit"] {
+ @apply border-rounded bg-cyan-900 px-2 cursor-pointer;
+ }
+
h1 {
@apply text-4xl;
}
diff --git a/src/truth-table.tsx b/src/truth-table.tsx
new file mode 100644
index 0000000..cd00b44
--- /dev/null
+++ b/src/truth-table.tsx
@@ -0,0 +1,355 @@
+/* @refresh reload */
+import Layout from "./components/layout";
+import { Input } from "./components/input";
+// import { Check, Download, Eye, EyeOff, Filter, Search, X } from "react-feather";
+import { Icon } from "solid-heroicons";
+import TruthTable from "./components/truth-table";
+import { InfoBox, MyDisclosure, MyDisclosureContainer } from "./components/output";
+// import MySwitch from "./components/switch";
+// import { diffChars } from "diff";
+// import { Menu } from "@headlessui/react";
+// import MyMenu from "./components/menu";
+// import { type BookType, utils, write, writeFile } from "xlsx"
+// import MyDialog from "./components/myDialog";
+import type { FetchResult } from "./types/interfaces";
+import { type Component, createSignal, JSX, Show } from "solid-js";
+import { For, render } from "solid-js/web";
+import Row from "./components/row";
+import { magnifyingGlass, xMark } from "solid-heroicons/solid";
+
+// TODO move some code to new components
+const TruthTablePage: Component = () => {
+
+ const inputId = "truth-input";
+
+ /**
+ * Stores the boolean value of the simplify toggle
+ */
+ const [simplifyEnabled, setSimplifyEnabled] = createSignal(true);
+ /**
+ * The state element used to store the simplified string, "empty string" by default
+ */
+ const [fetchResult, setFetchResult] = createSignal(null);
+
+ /**
+ * If the searchbar is empty, this state is 'false', otherwise 'true'
+ */
+ const [typing, setTyping] = createSignal(false);
+
+ const hideOptions = [
+ { name: "Show all result", value: "NONE" },
+ { name: "Hide true results", value: "TRUE" },
+ { name: "Hide false results", value: "FALSE" },
+ ];
+
+ const sortOptions = [
+ { name: "Sort by default", value: "DEFAULT" },
+ { name: "Sort by true first", value: "TRUE_FIRST" },
+ { name: "Sort by false first", value: "FALSE_FIRST" },
+ ];
+
+ /**
+ * The currently selected hide value, either 'none', 'true' or 'false'
+ */
+ const [hideValues, setHideValues] = createSignal(hideOptions[0]);
+
+ /**
+ * The currently selected sort value, either 'default', 'trueFirst' or 'falseFirst'
+ */
+ const [sortValues, setSortValues] = createSignal(sortOptions[0]);
+
+ /**
+ * Updates the state of the current expression to the new search with all whitespace removed.
+ * If the element is not found, reset.
+ */
+ async function onClick(e: { preventDefault: () => void; }): Promise {
+ e.preventDefault(); // Stops the page from reloading onClick
+ const exp = (document.getElementById(inputId) as HTMLInputElement | null)?.value;
+
+ if (exp && exp !== "") {
+
+ // TODO add loading animation
+ let result: FetchResult | undefined;
+ await fetch(`https://api.martials.no/simplify-truths/simplify/table?exp=${ exp }&simplify=${ simplifyEnabled() }`)
+ .then(res => res.json())
+ .then(res => result = res)
+ .catch(err => console.error(err)) // TODO show error on screen
+ .finally();
+
+ console.log(result);
+ setFetchResult(result);
+ }
+ else {
+ setFetchResult(null);
+ }
+ }
+
+ function onTyping() {
+ console.log("typing");
+ const el = (document.getElementById(inputId) as HTMLInputElement | null);
+ if (el && (el.value !== "") !== typing()) {
+ setTyping(el.value !== "");
+ }
+ }
+
+ function clearSearch() {
+ const el = (document.getElementById(inputId) as HTMLInputElement | null);
+ if (el) {
+ el.value = "";
+ setFetchResult(null);
+ setTyping(false);
+ el.focus();
+ }
+ }
+
+ const tableId = "truth-table";
+ const filenameId = "excel-filename";
+
+ document.addEventListener("DOMContentLoaded", () => {
+ // Focuses searchbar on load
+ (document.getElementById(inputId) as HTMLInputElement | null)?.focus();
+ });
+
+ /**
+ * Exports the generated truth table to an excel (.xlsx) file
+ *
+ * @param type The downloaded files extension. Default is "xlsx"
+ * @param name The name of the file, excluding the extension. Default is "Truth Table"
+ * @param dl
+ * @returns {any}
+ * @author SheetJS
+ * @link https://cdn.sheetjs.com/
+ * @license Apache 2.0 License
+ * SheetJS Community Edition -- https://sheetjs.com/
+ *
+ * Copyright (C) 2012-present SheetJS LLC
+ *
+ * Licensed under the Apache License, Version 2.0 (the "License");
+ * you may not use this file except in compliance with the License.
+ * You may obtain a copy of the License at
+ *
+ * http://www.apache.org/licenses/LICENSE-2.0
+ *
+ * Unless required by applicable law or agreed to in writing, software
+ * distributed under the License is distributed on an "AS IS" BASIS,
+ * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+ * See the License for the specific language governing permissions and
+ * limitations under the License.
+ */
+ // function exportToExcel(
+ // {
+ // type = "xlsx",
+ // name = "Truth Table",
+ // dl = false
+ // }: { type?: BookType, name?: string, dl?: boolean }): any {
+ //
+ // const element = document.getElementById(tableId);
+ // const wb = utils.table_to_book(element, { sheet: "sheet1" });
+ // return dl ?
+ // write(wb, { bookType: type, bookSST: true, type: 'base64' }) :
+ // writeFile(wb, name + "." + type);
+ // }
+ //
+ // function _exportToExcel(): void {
+ // const value = (document.getElementById(filenameId) as HTMLInputElement | null)?.value;
+ // exportToExcel({
+ // name: value !== "" ? value : undefined,
+ // });
+ // }
+
+ return (
+
+
+
+
+
+ { "truthTableHowTo" /*TODO*/ }
+
+
+ <>
+ not
+ and
+ or
+ implication
+ >
+
+
+
+
+
+
+ { "Simplify" }:
+ {/**/ }
+
+
+ {/* :*/ }
+ {/* // */ }
+ {/* // }*/ }
+ {/* children={*/ }
+ {/* */ }
+ {/* { (option) => (*/ }
+ {/* setHideValues(option) }*/ }
+ {/* option={ option }*/ }
+ {/* currentValue={ hideValues } />)*/ }
+ {/* }*/ }
+ {/* */ }
+ {/* } itemsClassName={ "right-0" }*/ }
+ {/*/>*/ }
+
+
+
+ {/* }*/ }
+ {/* children={*/ }
+ {/* */ }
+ {/* { option => (*/ }
+ {/* setSortValues(option) } />)*/ }
+ {/* }*/ }
+ {/* */ }
+ {/* }*/ }
+ {/* itemsClassName={ "right-0" }*/ }
+ {/*/>*/ }
+
+
+ {/*{*/ }
+ {/* fetchResult()?.expression &&*/ }
+ {/* { t("download") }
> }*/ }
+ {/* callback={ _exportToExcel }*/ }
+ {/* acceptButtonName={ t("download") }*/ }
+ {/* cancelButtonName={ t("cancel") }*/ }
+ {/* buttonClasses={ `float-right` }*/ }
+ {/* buttonTitle={ t("exportCurrentTable") }*/ }
+ {/* acceptButtonId={ "download-accept" }>*/ }
+ {/* { t("filename") }:
*/ }
+ {/* */ }
+ {/* */ }
+ {/*}*/ }
+
+
+ {/*{*/ }
+ {/* fetchResult && fetchResult()?.status.code !== 200 &&*/ }
+ {/*
*/ }
+ {/* { fetchResult()?.status.message }
*/ }
+ {/* */ }
+ {/*}*/ }
+ {/*{*/ }
+ {/* fetchResult()?.orderOperations && simplifyEnabled() && fetchResult()?.orderOperations.length > 0 &&*/ }
+ {/*
*/ }
+ {/* */ }
+ {/* */ }
+ {/* */ }
+ {/* {*/ }
+ {/* (operation, index) => (*/ }
+ {/* */ }
+ {/* { index() + 1 }: | */ }
+ {/* */ }
+ {/* /!**!/*/ }
+ {/* /!* { (part) => (*!/*/ }
+ {/* /!* *!/*/ }
+ {/* /!* { part.value }*!/*/ }
+ {/* /!* ) }*!/*/ }
+ {/* /!**!/*/ }
+ {/* { typeof window !== "undefined" && window.outerWidth <= 640 &&*/ }
+ {/* { t("using") }: { operation.law } }*/ }
+ {/* | */ }
+ {/* { typeof window !== "undefined" && window.outerWidth > 640 &&*/ }
+ {/* { t("using") }: { operation.law } | }*/ }
+ {/*
*/ }
+ {/* ) }*/ }
+ {/* */ }
+ {/* */ }
+ {/*
*/ }
+ {/* */ }
+ {/* */ }
+ {/*}*/ }
+
+ {
+ fetchResult()?.expression &&
+ <>
+ {/*
*/ }
+ {/* {*/ }
+ {/* simplifyEnabled &&*/ }
+ {/*
*/ }
+ {/* { fetchResult()?.after }
*/ }
+ {/* */ }
+ {/* }*/ }
+ {/*
*/ }
+
+
+ >
+ }
+
+
+ );
+}
+
+export default TruthTablePage;
+
+interface SingleMenuItem {
+ option: any,
+ currentValue?: any,
+ onClick: JSX.EventHandlerUnion,
+}
+
+const SingleMenuItem: Component = ({ option, currentValue, onClick }) => {
+ return (<>>
+ //
+ //
+ //
+ // { option.name }
+ //
+ //
+ );
+}
+
+render(() => , document.getElementById("root") as HTMLElement);
diff --git a/src/types/interfaces.ts b/src/types/interfaces.ts
index 25ec902..18aa369 100644
--- a/src/types/interfaces.ts
+++ b/src/types/interfaces.ts
@@ -1,6 +1,7 @@
import { JSX } from "solid-js";
export interface SimpleProps {
+ name?: string;
className?: string,
style?: JSX.CSSProperties,
id?: string,
@@ -20,6 +21,47 @@ export interface TitleProps extends ChildProps {
title?: string,
}
+export interface InputProps extends TitleProps {
+ onChange?: JSX.EventHandlerUnion,
+ placeholder?: string | null,
+ required?: boolean,
+ type?: string,
+}
+
export interface CardProps extends LinkProps {
title?: string;
-}
\ No newline at end of file
+}
+
+export type Expression = {
+ leading: string,
+ left: Expression | null,
+ operator: Operator | null,
+ right: Expression | null,
+ trailing: string,
+ atomic: string | null,
+};
+
+export type Operator = "AND" | "OR" | "NOT" | "IMPLICATION";
+
+export type Table = boolean[][];
+
+export type OrderOfOperations = {
+ before: string,
+ after: string,
+ law: string,
+}[];
+
+export type FetchResult = {
+ status: {
+ code: number,
+ message: string,
+ },
+ before: string,
+ after: string,
+ orderOperations: OrderOfOperations | null,
+ expression: Expression | null,
+ header: string[] | null,
+ table: {
+ truthMatrix: Table,
+ } | null,
+};
diff --git a/vite.config.ts b/vite.config.ts
index bd7292d..c50e23e 100644
--- a/vite.config.ts
+++ b/vite.config.ts
@@ -12,6 +12,7 @@ export default defineConfig({
input: {
main: "index.html",
"404": "404.html",
+ simplifyTruths: "simplify-truths.html",
}
}
},