galadrim-feedback 0.0.2 → 0.0.4
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/dist/index.cjs.js +2 -11
- package/dist/index.es.js +2 -11
- package/dist/src/index.css +1 -1
- package/package.json +7 -1
- package/eslint.config.js +0 -28
- package/rollup.config.js +0 -40
- package/src/App.tsx +0 -16
- package/src/components/buttons/IconButton.tsx +0 -27
- package/src/components/buttons/PlusButton.tsx +0 -20
- package/src/components/feedback/FeedbackItem.tsx +0 -89
- package/src/components/misc/Avatar.tsx +0 -3
- package/src/components/misc/Modal.tsx +0 -43
- package/src/index.css +0 -7
- package/src/pages/FeedbacksCanvas.tsx +0 -38
- package/src/pages/Overlay.tsx +0 -26
- package/src/pages/Root.tsx +0 -128
- package/src/services/api.ts +0 -22
- package/src/services/feedback.ts +0 -62
- package/src/types/types.d.ts +0 -16
- package/tailwind.config.js +0 -46
- package/tsconfig.json +0 -18
package/dist/index.cjs.js
CHANGED
|
@@ -3,8 +3,8 @@
|
|
|
3
3
|
var jsxRuntime = require('react/jsx-runtime');
|
|
4
4
|
var reactQuery = require('@tanstack/react-query');
|
|
5
5
|
var React = require('react');
|
|
6
|
-
var axios = require('axios');
|
|
7
6
|
var reactDom = require('react-dom');
|
|
7
|
+
var axios = require('axios');
|
|
8
8
|
|
|
9
9
|
const api = axios.create({
|
|
10
10
|
baseURL: "http://localhost:3000",
|
|
@@ -112,16 +112,13 @@ const FeedbackItem = ({ feedback }) => {
|
|
|
112
112
|
}, children: [jsxRuntime.jsxs("div", { children: [jsxRuntime.jsx("p", { style: { fontSize: "16px", fontWeight: "bold" }, children: "Feedback" }), jsxRuntime.jsxs("p", { style: { fontSize: "14px" }, children: ["Created at: ", feedback.createdAt.toLocaleString()] }), jsxRuntime.jsx("a", { href: feedback.elementId, target: "_blank", children: "Go to component" })] }), jsxRuntime.jsx("button", { onClick: handleDeleteFeedback, children: "Delete" })] }), jsxRuntime.jsx("p", { style: { fontSize: "14px" }, children: feedback.comment })] }), document.body);
|
|
113
113
|
};
|
|
114
114
|
|
|
115
|
-
new reactQuery.QueryClient();
|
|
116
115
|
function Root() {
|
|
117
116
|
const [isOpen, setIsOpen] = React.useState(false);
|
|
118
117
|
const { data: feedbacks } = useFeedbacks();
|
|
119
118
|
console.log("feedbacks 1 : ", feedbacks);
|
|
120
|
-
const [creatingFeedback, setCreatingFeedback] = React.useState(false);
|
|
121
119
|
const { mutate: createFeedback } = useCreateFeedback();
|
|
122
120
|
React.useEffect(() => {
|
|
123
121
|
if (isOpen) {
|
|
124
|
-
console.log("Opening feedbacks canvas");
|
|
125
122
|
window.addEventListener("keydown", handleKeyPress);
|
|
126
123
|
return () => {
|
|
127
124
|
window.removeEventListener("keypress", handleKeyPress);
|
|
@@ -129,13 +126,11 @@ function Root() {
|
|
|
129
126
|
}
|
|
130
127
|
}, [isOpen]);
|
|
131
128
|
const handleKeyPress = (event) => {
|
|
132
|
-
console.log(event.key);
|
|
133
129
|
if (event.key === "Escape") {
|
|
134
130
|
setIsOpen(false);
|
|
135
131
|
}
|
|
136
132
|
};
|
|
137
133
|
const handleClick = (e) => {
|
|
138
|
-
console.log("click");
|
|
139
134
|
e.stopPropagation();
|
|
140
135
|
e.preventDefault();
|
|
141
136
|
const node = e.target;
|
|
@@ -149,8 +144,6 @@ function Root() {
|
|
|
149
144
|
const newFeedback = {
|
|
150
145
|
x,
|
|
151
146
|
y,
|
|
152
|
-
screenWidth: window.innerWidth,
|
|
153
|
-
screenHeight: window.innerHeight,
|
|
154
147
|
comment: "test",
|
|
155
148
|
path: window.location.pathname,
|
|
156
149
|
elementId,
|
|
@@ -160,12 +153,10 @@ function Root() {
|
|
|
160
153
|
createFeedback(newFeedback);
|
|
161
154
|
document.body.classList.remove("comment-cursor");
|
|
162
155
|
document.removeEventListener("click", handleClick, true);
|
|
163
|
-
setCreatingFeedback(false);
|
|
164
156
|
setIsOpen(true);
|
|
165
157
|
};
|
|
166
158
|
const handleCreateFeedback = (e) => {
|
|
167
159
|
e.stopPropagation();
|
|
168
|
-
setCreatingFeedback(true);
|
|
169
160
|
document.body.classList.add("comment-cursor");
|
|
170
161
|
document.addEventListener("click", handleClick, true);
|
|
171
162
|
};
|
|
@@ -202,4 +193,4 @@ function App() {
|
|
|
202
193
|
}
|
|
203
194
|
|
|
204
195
|
module.exports = App;
|
|
205
|
-
//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"file":"index.cjs.js","sources":["../src/services/api.ts","../src/services/feedback.ts","../src/components/feedback/FeedbackItem.tsx","../src/pages/Root.tsx","../src/App.tsx"],"sourcesContent":["import axios from \"axios\";\n\nconst api = axios.create({\n  baseURL: \"http://localhost:3000\",\n});\n\napi.interceptors.request.use(\n  async (config) => {\n    // const token = localStorage.getItem(\"dashboardGaladrimAuthToken\");\n    // if (token && config.headers) {\n    //   config.headers[\"x-dashboard-token\"] = token;\n    // } else {\n    //   delete api.defaults.headers.common.Authorization;\n    // }\n    return config;\n  },\n  (error) => {\n    return Promise.reject(error);\n  }\n);\n\nexport default api;\n","import { useMutation, useQuery, useQueryClient } from \"@tanstack/react-query\";\nimport api from \"./api\";\nimport { Feedback, FeedbackPayload } from \"../types/types\";\n\nexport const fetchFeedbacks = async () => {\n  const response = await api.get<Feedback[]>(\"/feedback\");\n  return response.data;\n};\n\nexport const createFeedback = async (feedback: FeedbackPayload) => {\n  const response = await api.post(\"/feedback\", feedback);\n  return response.data;\n};\n\nexport const updateFeedback = async (feedback: FeedbackPayload) => {\n  const response = await api.put(\"/feedback\", feedback);\n  return response.data;\n};\n\nexport const deleteFeedback = async (feedback: Feedback) => {\n  const response = await api.delete(`/feedback/${feedback.id}`);\n  return response.data;\n};\n\nexport const useFeedbacks = () => {\n  return useQuery({\n    queryKey: [\"feedbacks\"],\n    queryFn: fetchFeedbacks,\n  });\n};\n\nexport const useCreateFeedback = () => {\n  const queryClient = useQueryClient();\n  return useMutation({\n    mutationFn: createFeedback,\n    onSuccess: () => {\n      console.log(\"Feedback created\");\n\n      return queryClient.invalidateQueries({ queryKey: [\"feedbacks\"] });\n    },\n  });\n};\n\nexport const useUpdateFeedback = () => {\n  const queryClient = useQueryClient();\n  return useMutation({\n    mutationFn: updateFeedback,\n    onSuccess: () => {\n      return queryClient.invalidateQueries({ queryKey: [\"feedbacks\"] });\n    },\n  });\n};\n\nexport const useDeleteFeedback = () => {\n  const queryClient = useQueryClient();\n  return useMutation({\n    mutationFn: deleteFeedback,\n    onSuccess: () => {\n      return queryClient.invalidateQueries({ queryKey: [\"feedbacks\"] });\n    },\n  });\n};\n","import { createPortal } from \"react-dom\";\nimport { useEffect, useRef, useState } from \"react\";\nimport { Feedback } from \"../../types/types\";\nimport { useDeleteFeedback } from \"../../services/feedback\";\n\nexport const FeedbackItem = ({ feedback }: { feedback: Feedback }) => {\n  const targetElement = useRef<HTMLElement | null>(null);\n  const { mutate: deleteFeedback } = useDeleteFeedback();\n  const [position, setPosition] = useState({ top: 0, left: 0 });\n\n  useEffect(() => {\n    const handleResize = () => {\n      if (!targetElement || !targetElement.current) {\n        return;\n      }\n      const dimensions = targetElement.current.getBoundingClientRect();\n      setPosition({\n        top: feedback.y + dimensions.top + window.scrollY,\n        left: feedback.x + dimensions.left + window.scrollX,\n      });\n    };\n\n    const elements = document.querySelectorAll<HTMLElement>(\n      `[data-id=\"${feedback.elementId}\"]`\n    );\n    const element = elements[feedback.elementIndex];\n\n    if (element) {\n      console.log(element);\n\n      targetElement.current = element;\n      handleResize();\n      window.addEventListener(\"resize\", handleResize);\n    }\n\n    return () => {\n      if (element) {\n        window.removeEventListener(\"resize\", handleResize);\n      }\n    };\n  }, [feedback]);\n\n  const handleDeleteFeedback = () => {\n    deleteFeedback(feedback);\n  };\n\n  // If we haven't found a matching DOM node, don't render anything\n  if (!targetElement || !targetElement.current) {\n    return null;\n  }\n\n  console.log(\"rendrr\", position);\n\n  // Render this component inside `targetElement` using a Portal\n  return createPortal(\n    <div\n      style={{\n        position: \"absolute\",\n        top: position.top,\n        left: position.left,\n        width: \"200px\",\n        height: \"100px\",\n        border: \"1px solid black\",\n        backgroundColor: \"white\",\n      }}\n    >\n      <div\n        style={{\n          display: \"flex\",\n          justifyContent: \"space-between\",\n          alignItems: \"center\",\n        }}\n      >\n        <div>\n          <p style={{ fontSize: \"16px\", fontWeight: \"bold\" }}>Feedback</p>\n          <p style={{ fontSize: \"14px\" }}>\n            Created at: {feedback.createdAt.toLocaleString()}\n          </p>\n          <a href={feedback.elementId} target=\"_blank\">\n            Go to component\n          </a>\n        </div>\n        <button onClick={handleDeleteFeedback}>Delete</button>\n      </div>\n      <p style={{ fontSize: \"14px\" }}>{feedback.comment}</p>\n    </div>,\n    document.body\n  );\n};\n","import { QueryClient } from \"@tanstack/react-query\";\nimport React, { useEffect } from \"react\";\nimport ReactDOM from \"react-dom\";\nimport { Overlay } from \"./Overlay\";\nimport { useCreateFeedback, useFeedbacks } from \"../services/feedback\";\nimport { FeedbackPayload } from \"../types/types\";\nimport { FeedbackItem } from \"../components/feedback/FeedbackItem\";\n\nconst queryClient = new QueryClient();\n\nfunction Root() {\n  const [isOpen, setIsOpen] = React.useState(false);\n  const { data: feedbacks } = useFeedbacks();\n\n  console.log(\"feedbacks 1 : \", feedbacks);\n\n  const [creatingFeedback, setCreatingFeedback] = React.useState(false);\n\n  const { mutate: createFeedback } = useCreateFeedback();\n\n  useEffect(() => {\n    if (isOpen) {\n      console.log(\"Opening feedbacks canvas\");\n\n      window.addEventListener(\"keydown\", handleKeyPress);\n      return () => {\n        window.removeEventListener(\"keypress\", handleKeyPress);\n      };\n    }\n  }, [isOpen]);\n\n  const handleKeyPress = (event: KeyboardEvent) => {\n    console.log(event.key);\n\n    if (event.key === \"Escape\") {\n      setIsOpen(false);\n    }\n  };\n\n  const handleClick = (e: any) => {\n    console.log(\"click\");\n\n    e.stopPropagation();\n    e.preventDefault();\n    const node = e.target as HTMLElement | null;\n    const elementId = node?.dataset.id || \"\";\n    const allElements = document.querySelectorAll<HTMLElement>(\n      `[data-id=\"${elementId}\"]`\n    );\n\n    const arrAllElements = Array.from(allElements);\n\n    const elementIndex = arrAllElements.indexOf(node as HTMLElement);\n\n    const x = e.offsetX;\n    const y = e.offsetY;\n    console.log(e);\n    const newFeedback: FeedbackPayload = {\n      x,\n      y,\n      screenWidth: window.innerWidth,\n      screenHeight: window.innerHeight,\n      comment: \"test\",\n      path: window.location.pathname,\n      elementId,\n      elementIndex,\n    };\n    console.log(newFeedback);\n    createFeedback(newFeedback);\n    document.body.classList.remove(\"comment-cursor\");\n    document.removeEventListener(\"click\", handleClick, true);\n    setCreatingFeedback(false);\n    setIsOpen(true);\n  };\n\n  const handleCreateFeedback = (e: React.MouseEvent<HTMLButtonElement>) => {\n    e.stopPropagation();\n    setCreatingFeedback(true);\n    document.body.classList.add(\"comment-cursor\");\n    document.addEventListener(\"click\", handleClick, true);\n  };\n\n  return (\n    <>\n      <div\n        style={{\n          position: \"fixed\",\n          top: \"calc(100vh - 62px)\",\n          left: \"calc(100vw - 124px)\",\n          width: \"100px\",\n          height: \"50px\",\n          zIndex: 1000000,\n          backgroundColor: \"#0085FF\",\n          color: \"white\",\n          borderRadius: 4,\n        }}\n        onClick={() => setIsOpen(!isOpen)}\n      >\n        Show Feedback\n      </div>\n      <button\n        style={{\n          position: \"fixed\",\n          top: \"calc(100vh - 122px)\",\n          left: \"calc(100vw - 124px)\",\n          width: \"100px\",\n          height: \"50px\",\n          zIndex: 1000000,\n          backgroundColor: \"#0085FF\",\n          color: \"white\",\n          borderRadius: 4,\n        }}\n        onClick={handleCreateFeedback}\n      >\n        New Feedback\n      </button>\n      {isOpen &&\n        feedbacks &&\n        feedbacks\n          .filter((feedback) => feedback.path === window.location.pathname)\n          .map((feedback) => (\n            <FeedbackItem key={feedback.id} feedback={feedback} />\n          ))}\n    </>\n  );\n}\n\nexport default Root;\n","import { QueryClient, QueryClientProvider } from \"@tanstack/react-query\";\nimport Root from \"./pages/Root\";\nimport \"./index.css\";\nimport React from \"react\";\n\nconst queryClient = new QueryClient();\n\nfunction App() {\n  return (\n    <QueryClientProvider client={queryClient}>\n      <Root />\n    </QueryClientProvider>\n  );\n}\n\nexport default App;\n"],"names":["useQuery","useQueryClient","useMutation","useRef","useState","useEffect","createPortal","_jsxs","_jsx","QueryClient","_Fragment","QueryClientProvider"],"mappings":"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"}
|
|
196
|
+
//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"file":"index.cjs.js","sources":["../src/services/api.ts","../src/services/feedback.ts","../src/components/feedback/FeedbackItem.tsx","../src/pages/Root.tsx","../src/App.tsx"],"sourcesContent":["import axios from \"axios\";\n\nconst api = axios.create({\n  baseURL: \"http://localhost:3000\",\n});\n\napi.interceptors.request.use(\n  async (config) => {\n    // const token = localStorage.getItem(\"dashboardGaladrimAuthToken\");\n    // if (token && config.headers) {\n    //   config.headers[\"x-dashboard-token\"] = token;\n    // } else {\n    //   delete api.defaults.headers.common.Authorization;\n    // }\n    return config;\n  },\n  (error) => {\n    return Promise.reject(error);\n  }\n);\n\nexport default api;\n","import { useMutation, useQuery, useQueryClient } from \"@tanstack/react-query\";\nimport api from \"./api\";\nimport { Feedback, FeedbackPayload } from \"../types/types\";\n\nexport const fetchFeedbacks = async () => {\n  const response = await api.get<Feedback[]>(\"/feedback\");\n  return response.data;\n};\n\nexport const createFeedback = async (feedback: FeedbackPayload) => {\n  const response = await api.post(\"/feedback\", feedback);\n  return response.data;\n};\n\nexport const updateFeedback = async (feedback: FeedbackPayload) => {\n  const response = await api.put(\"/feedback\", feedback);\n  return response.data;\n};\n\nexport const deleteFeedback = async (feedback: Feedback) => {\n  const response = await api.delete(`/feedback/${feedback.id}`);\n  return response.data;\n};\n\nexport const useFeedbacks = () => {\n  return useQuery({\n    queryKey: [\"feedbacks\"],\n    queryFn: fetchFeedbacks,\n  });\n};\n\nexport const useCreateFeedback = () => {\n  const queryClient = useQueryClient();\n  return useMutation({\n    mutationFn: createFeedback,\n    onSuccess: () => {\n      console.log(\"Feedback created\");\n\n      return queryClient.invalidateQueries({ queryKey: [\"feedbacks\"] });\n    },\n  });\n};\n\nexport const useUpdateFeedback = () => {\n  const queryClient = useQueryClient();\n  return useMutation({\n    mutationFn: updateFeedback,\n    onSuccess: () => {\n      return queryClient.invalidateQueries({ queryKey: [\"feedbacks\"] });\n    },\n  });\n};\n\nexport const useDeleteFeedback = () => {\n  const queryClient = useQueryClient();\n  return useMutation({\n    mutationFn: deleteFeedback,\n    onSuccess: () => {\n      return queryClient.invalidateQueries({ queryKey: [\"feedbacks\"] });\n    },\n  });\n};\n","import { createPortal } from \"react-dom\";\nimport { useEffect, useRef, useState } from \"react\";\nimport { Feedback } from \"../../types/types\";\nimport { useDeleteFeedback } from \"../../services/feedback\";\n\nexport const FeedbackItem = ({ feedback }: { feedback: Feedback }) => {\n  const targetElement = useRef<HTMLElement | null>(null);\n  const { mutate: deleteFeedback } = useDeleteFeedback();\n  const [position, setPosition] = useState({ top: 0, left: 0 });\n\n  useEffect(() => {\n    const handleResize = () => {\n      if (!targetElement || !targetElement.current) {\n        return;\n      }\n      const dimensions = targetElement.current.getBoundingClientRect();\n      setPosition({\n        top: feedback.y + dimensions.top + window.scrollY,\n        left: feedback.x + dimensions.left + window.scrollX,\n      });\n    };\n\n    const elements = document.querySelectorAll<HTMLElement>(\n      `[data-id=\"${feedback.elementId}\"]`\n    );\n    const element = elements[feedback.elementIndex];\n\n    if (element) {\n      console.log(element);\n\n      targetElement.current = element;\n      handleResize();\n      window.addEventListener(\"resize\", handleResize);\n    }\n\n    return () => {\n      if (element) {\n        window.removeEventListener(\"resize\", handleResize);\n      }\n    };\n  }, [feedback]);\n\n  const handleDeleteFeedback = () => {\n    deleteFeedback(feedback);\n  };\n\n  // If we haven't found a matching DOM node, don't render anything\n  if (!targetElement || !targetElement.current) {\n    return null;\n  }\n\n  console.log(\"rendrr\", position);\n\n  // Render this component inside `targetElement` using a Portal\n  return createPortal(\n    <div\n      style={{\n        position: \"absolute\",\n        top: position.top,\n        left: position.left,\n        width: \"200px\",\n        height: \"100px\",\n        border: \"1px solid black\",\n        backgroundColor: \"white\",\n      }}\n    >\n      <div\n        style={{\n          display: \"flex\",\n          justifyContent: \"space-between\",\n          alignItems: \"center\",\n        }}\n      >\n        <div>\n          <p style={{ fontSize: \"16px\", fontWeight: \"bold\" }}>Feedback</p>\n          <p style={{ fontSize: \"14px\" }}>\n            Created at: {feedback.createdAt.toLocaleString()}\n          </p>\n          <a href={feedback.elementId} target=\"_blank\">\n            Go to component\n          </a>\n        </div>\n        <button onClick={handleDeleteFeedback}>Delete</button>\n      </div>\n      <p style={{ fontSize: \"14px\" }}>{feedback.comment}</p>\n    </div>,\n    document.body\n  );\n};\n","import React, { useEffect } from \"react\";\nimport { FeedbackItem } from \"../components/feedback/FeedbackItem\";\nimport { useCreateFeedback, useFeedbacks } from \"../services/feedback\";\nimport { FeedbackPayload } from \"../types/types\";\n\nfunction Root() {\n  const [isOpen, setIsOpen] = React.useState(false);\n  const { data: feedbacks } = useFeedbacks();\n\n  console.log(\"feedbacks 1 : \", feedbacks);\n\n  const { mutate: createFeedback } = useCreateFeedback();\n\n  useEffect(() => {\n    if (isOpen) {\n      window.addEventListener(\"keydown\", handleKeyPress);\n      return () => {\n        window.removeEventListener(\"keypress\", handleKeyPress);\n      };\n    }\n  }, [isOpen]);\n\n  const handleKeyPress = (event: KeyboardEvent) => {\n    if (event.key === \"Escape\") {\n      setIsOpen(false);\n    }\n  };\n\n  const handleClick = (e: MouseEvent) => {\n    e.stopPropagation();\n    e.preventDefault();\n    const node = e.target as HTMLElement | null;\n    const elementId = node?.dataset.id || \"\";\n    const allElements = document.querySelectorAll<HTMLElement>(\n      `[data-id=\"${elementId}\"]`\n    );\n\n    const arrAllElements = Array.from(allElements);\n\n    const elementIndex = arrAllElements.indexOf(node as HTMLElement);\n\n    const x = e.offsetX;\n    const y = e.offsetY;\n    console.log(e);\n    const newFeedback: FeedbackPayload = {\n      x,\n      y,\n      comment: \"test\",\n      path: window.location.pathname,\n      elementId,\n      elementIndex,\n    };\n    console.log(newFeedback);\n    createFeedback(newFeedback);\n    document.body.classList.remove(\"comment-cursor\");\n    document.removeEventListener(\"click\", handleClick, true);\n    setIsOpen(true);\n  };\n\n  const handleCreateFeedback = (e: React.MouseEvent<HTMLButtonElement>) => {\n    e.stopPropagation();\n    document.body.classList.add(\"comment-cursor\");\n    document.addEventListener(\"click\", handleClick, true);\n  };\n\n  return (\n    <>\n      <div\n        style={{\n          position: \"fixed\",\n          top: \"calc(100vh - 62px)\",\n          left: \"calc(100vw - 124px)\",\n          width: \"100px\",\n          height: \"50px\",\n          zIndex: 1000000,\n          backgroundColor: \"#0085FF\",\n          color: \"white\",\n          borderRadius: 4,\n        }}\n        onClick={() => setIsOpen(!isOpen)}\n      >\n        Show Feedback\n      </div>\n      <button\n        style={{\n          position: \"fixed\",\n          top: \"calc(100vh - 122px)\",\n          left: \"calc(100vw - 124px)\",\n          width: \"100px\",\n          height: \"50px\",\n          zIndex: 1000000,\n          backgroundColor: \"#0085FF\",\n          color: \"white\",\n          borderRadius: 4,\n        }}\n        onClick={handleCreateFeedback}\n      >\n        New Feedback\n      </button>\n      {isOpen &&\n        feedbacks &&\n        feedbacks\n          .filter((feedback) => feedback.path === window.location.pathname)\n          .map((feedback) => (\n            <FeedbackItem key={feedback.id} feedback={feedback} />\n          ))}\n    </>\n  );\n}\n\nexport default Root;\n","import { QueryClient, QueryClientProvider } from \"@tanstack/react-query\";\nimport Root from \"./pages/Root\";\nimport \"./index.css\";\nimport React from \"react\";\n\nconst queryClient = new QueryClient();\n\nfunction App() {\n  return (\n    <QueryClientProvider client={queryClient}>\n      <Root />\n    </QueryClientProvider>\n  );\n}\n\nexport default App;\n"],"names":["useQuery","useQueryClient","useMutation","useRef","useState","useEffect","createPortal","_jsxs","_jsx","_Fragment","QueryClient","QueryClientProvider"],"mappings":"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"}
|
package/dist/index.es.js
CHANGED
|
@@ -1,8 +1,8 @@
|
|
|
1
1
|
import { jsxs, jsx, Fragment } from 'react/jsx-runtime';
|
|
2
2
|
import { useQuery, useQueryClient, useMutation, QueryClient, QueryClientProvider } from '@tanstack/react-query';
|
|
3
3
|
import React, { useRef, useState, useEffect } from 'react';
|
|
4
|
-
import axios from 'axios';
|
|
5
4
|
import { createPortal } from 'react-dom';
|
|
5
|
+
import axios from 'axios';
|
|
6
6
|
|
|
7
7
|
const api = axios.create({
|
|
8
8
|
baseURL: "http://localhost:3000",
|
|
@@ -110,16 +110,13 @@ const FeedbackItem = ({ feedback }) => {
|
|
|
110
110
|
}, children: [jsxs("div", { children: [jsx("p", { style: { fontSize: "16px", fontWeight: "bold" }, children: "Feedback" }), jsxs("p", { style: { fontSize: "14px" }, children: ["Created at: ", feedback.createdAt.toLocaleString()] }), jsx("a", { href: feedback.elementId, target: "_blank", children: "Go to component" })] }), jsx("button", { onClick: handleDeleteFeedback, children: "Delete" })] }), jsx("p", { style: { fontSize: "14px" }, children: feedback.comment })] }), document.body);
|
|
111
111
|
};
|
|
112
112
|
|
|
113
|
-
new QueryClient();
|
|
114
113
|
function Root() {
|
|
115
114
|
const [isOpen, setIsOpen] = React.useState(false);
|
|
116
115
|
const { data: feedbacks } = useFeedbacks();
|
|
117
116
|
console.log("feedbacks 1 : ", feedbacks);
|
|
118
|
-
const [creatingFeedback, setCreatingFeedback] = React.useState(false);
|
|
119
117
|
const { mutate: createFeedback } = useCreateFeedback();
|
|
120
118
|
useEffect(() => {
|
|
121
119
|
if (isOpen) {
|
|
122
|
-
console.log("Opening feedbacks canvas");
|
|
123
120
|
window.addEventListener("keydown", handleKeyPress);
|
|
124
121
|
return () => {
|
|
125
122
|
window.removeEventListener("keypress", handleKeyPress);
|
|
@@ -127,13 +124,11 @@ function Root() {
|
|
|
127
124
|
}
|
|
128
125
|
}, [isOpen]);
|
|
129
126
|
const handleKeyPress = (event) => {
|
|
130
|
-
console.log(event.key);
|
|
131
127
|
if (event.key === "Escape") {
|
|
132
128
|
setIsOpen(false);
|
|
133
129
|
}
|
|
134
130
|
};
|
|
135
131
|
const handleClick = (e) => {
|
|
136
|
-
console.log("click");
|
|
137
132
|
e.stopPropagation();
|
|
138
133
|
e.preventDefault();
|
|
139
134
|
const node = e.target;
|
|
@@ -147,8 +142,6 @@ function Root() {
|
|
|
147
142
|
const newFeedback = {
|
|
148
143
|
x,
|
|
149
144
|
y,
|
|
150
|
-
screenWidth: window.innerWidth,
|
|
151
|
-
screenHeight: window.innerHeight,
|
|
152
145
|
comment: "test",
|
|
153
146
|
path: window.location.pathname,
|
|
154
147
|
elementId,
|
|
@@ -158,12 +151,10 @@ function Root() {
|
|
|
158
151
|
createFeedback(newFeedback);
|
|
159
152
|
document.body.classList.remove("comment-cursor");
|
|
160
153
|
document.removeEventListener("click", handleClick, true);
|
|
161
|
-
setCreatingFeedback(false);
|
|
162
154
|
setIsOpen(true);
|
|
163
155
|
};
|
|
164
156
|
const handleCreateFeedback = (e) => {
|
|
165
157
|
e.stopPropagation();
|
|
166
|
-
setCreatingFeedback(true);
|
|
167
158
|
document.body.classList.add("comment-cursor");
|
|
168
159
|
document.addEventListener("click", handleClick, true);
|
|
169
160
|
};
|
|
@@ -200,4 +191,4 @@ function App() {
|
|
|
200
191
|
}
|
|
201
192
|
|
|
202
193
|
export { App as default };
|
|
203
|
-
//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"file":"index.es.js","sources":["../src/services/api.ts","../src/services/feedback.ts","../src/components/feedback/FeedbackItem.tsx","../src/pages/Root.tsx","../src/App.tsx"],"sourcesContent":["import axios from \"axios\";\n\nconst api = axios.create({\n  baseURL: \"http://localhost:3000\",\n});\n\napi.interceptors.request.use(\n  async (config) => {\n    // const token = localStorage.getItem(\"dashboardGaladrimAuthToken\");\n    // if (token && config.headers) {\n    //   config.headers[\"x-dashboard-token\"] = token;\n    // } else {\n    //   delete api.defaults.headers.common.Authorization;\n    // }\n    return config;\n  },\n  (error) => {\n    return Promise.reject(error);\n  }\n);\n\nexport default api;\n","import { useMutation, useQuery, useQueryClient } from \"@tanstack/react-query\";\nimport api from \"./api\";\nimport { Feedback, FeedbackPayload } from \"../types/types\";\n\nexport const fetchFeedbacks = async () => {\n  const response = await api.get<Feedback[]>(\"/feedback\");\n  return response.data;\n};\n\nexport const createFeedback = async (feedback: FeedbackPayload) => {\n  const response = await api.post(\"/feedback\", feedback);\n  return response.data;\n};\n\nexport const updateFeedback = async (feedback: FeedbackPayload) => {\n  const response = await api.put(\"/feedback\", feedback);\n  return response.data;\n};\n\nexport const deleteFeedback = async (feedback: Feedback) => {\n  const response = await api.delete(`/feedback/${feedback.id}`);\n  return response.data;\n};\n\nexport const useFeedbacks = () => {\n  return useQuery({\n    queryKey: [\"feedbacks\"],\n    queryFn: fetchFeedbacks,\n  });\n};\n\nexport const useCreateFeedback = () => {\n  const queryClient = useQueryClient();\n  return useMutation({\n    mutationFn: createFeedback,\n    onSuccess: () => {\n      console.log(\"Feedback created\");\n\n      return queryClient.invalidateQueries({ queryKey: [\"feedbacks\"] });\n    },\n  });\n};\n\nexport const useUpdateFeedback = () => {\n  const queryClient = useQueryClient();\n  return useMutation({\n    mutationFn: updateFeedback,\n    onSuccess: () => {\n      return queryClient.invalidateQueries({ queryKey: [\"feedbacks\"] });\n    },\n  });\n};\n\nexport const useDeleteFeedback = () => {\n  const queryClient = useQueryClient();\n  return useMutation({\n    mutationFn: deleteFeedback,\n    onSuccess: () => {\n      return queryClient.invalidateQueries({ queryKey: [\"feedbacks\"] });\n    },\n  });\n};\n","import { createPortal } from \"react-dom\";\nimport { useEffect, useRef, useState } from \"react\";\nimport { Feedback } from \"../../types/types\";\nimport { useDeleteFeedback } from \"../../services/feedback\";\n\nexport const FeedbackItem = ({ feedback }: { feedback: Feedback }) => {\n  const targetElement = useRef<HTMLElement | null>(null);\n  const { mutate: deleteFeedback } = useDeleteFeedback();\n  const [position, setPosition] = useState({ top: 0, left: 0 });\n\n  useEffect(() => {\n    const handleResize = () => {\n      if (!targetElement || !targetElement.current) {\n        return;\n      }\n      const dimensions = targetElement.current.getBoundingClientRect();\n      setPosition({\n        top: feedback.y + dimensions.top + window.scrollY,\n        left: feedback.x + dimensions.left + window.scrollX,\n      });\n    };\n\n    const elements = document.querySelectorAll<HTMLElement>(\n      `[data-id=\"${feedback.elementId}\"]`\n    );\n    const element = elements[feedback.elementIndex];\n\n    if (element) {\n      console.log(element);\n\n      targetElement.current = element;\n      handleResize();\n      window.addEventListener(\"resize\", handleResize);\n    }\n\n    return () => {\n      if (element) {\n        window.removeEventListener(\"resize\", handleResize);\n      }\n    };\n  }, [feedback]);\n\n  const handleDeleteFeedback = () => {\n    deleteFeedback(feedback);\n  };\n\n  // If we haven't found a matching DOM node, don't render anything\n  if (!targetElement || !targetElement.current) {\n    return null;\n  }\n\n  console.log(\"rendrr\", position);\n\n  // Render this component inside `targetElement` using a Portal\n  return createPortal(\n    <div\n      style={{\n        position: \"absolute\",\n        top: position.top,\n        left: position.left,\n        width: \"200px\",\n        height: \"100px\",\n        border: \"1px solid black\",\n        backgroundColor: \"white\",\n      }}\n    >\n      <div\n        style={{\n          display: \"flex\",\n          justifyContent: \"space-between\",\n          alignItems: \"center\",\n        }}\n      >\n        <div>\n          <p style={{ fontSize: \"16px\", fontWeight: \"bold\" }}>Feedback</p>\n          <p style={{ fontSize: \"14px\" }}>\n            Created at: {feedback.createdAt.toLocaleString()}\n          </p>\n          <a href={feedback.elementId} target=\"_blank\">\n            Go to component\n          </a>\n        </div>\n        <button onClick={handleDeleteFeedback}>Delete</button>\n      </div>\n      <p style={{ fontSize: \"14px\" }}>{feedback.comment}</p>\n    </div>,\n    document.body\n  );\n};\n","import { QueryClient } from \"@tanstack/react-query\";\nimport React, { useEffect } from \"react\";\nimport ReactDOM from \"react-dom\";\nimport { Overlay } from \"./Overlay\";\nimport { useCreateFeedback, useFeedbacks } from \"../services/feedback\";\nimport { FeedbackPayload } from \"../types/types\";\nimport { FeedbackItem } from \"../components/feedback/FeedbackItem\";\n\nconst queryClient = new QueryClient();\n\nfunction Root() {\n  const [isOpen, setIsOpen] = React.useState(false);\n  const { data: feedbacks } = useFeedbacks();\n\n  console.log(\"feedbacks 1 : \", feedbacks);\n\n  const [creatingFeedback, setCreatingFeedback] = React.useState(false);\n\n  const { mutate: createFeedback } = useCreateFeedback();\n\n  useEffect(() => {\n    if (isOpen) {\n      console.log(\"Opening feedbacks canvas\");\n\n      window.addEventListener(\"keydown\", handleKeyPress);\n      return () => {\n        window.removeEventListener(\"keypress\", handleKeyPress);\n      };\n    }\n  }, [isOpen]);\n\n  const handleKeyPress = (event: KeyboardEvent) => {\n    console.log(event.key);\n\n    if (event.key === \"Escape\") {\n      setIsOpen(false);\n    }\n  };\n\n  const handleClick = (e: any) => {\n    console.log(\"click\");\n\n    e.stopPropagation();\n    e.preventDefault();\n    const node = e.target as HTMLElement | null;\n    const elementId = node?.dataset.id || \"\";\n    const allElements = document.querySelectorAll<HTMLElement>(\n      `[data-id=\"${elementId}\"]`\n    );\n\n    const arrAllElements = Array.from(allElements);\n\n    const elementIndex = arrAllElements.indexOf(node as HTMLElement);\n\n    const x = e.offsetX;\n    const y = e.offsetY;\n    console.log(e);\n    const newFeedback: FeedbackPayload = {\n      x,\n      y,\n      screenWidth: window.innerWidth,\n      screenHeight: window.innerHeight,\n      comment: \"test\",\n      path: window.location.pathname,\n      elementId,\n      elementIndex,\n    };\n    console.log(newFeedback);\n    createFeedback(newFeedback);\n    document.body.classList.remove(\"comment-cursor\");\n    document.removeEventListener(\"click\", handleClick, true);\n    setCreatingFeedback(false);\n    setIsOpen(true);\n  };\n\n  const handleCreateFeedback = (e: React.MouseEvent<HTMLButtonElement>) => {\n    e.stopPropagation();\n    setCreatingFeedback(true);\n    document.body.classList.add(\"comment-cursor\");\n    document.addEventListener(\"click\", handleClick, true);\n  };\n\n  return (\n    <>\n      <div\n        style={{\n          position: \"fixed\",\n          top: \"calc(100vh - 62px)\",\n          left: \"calc(100vw - 124px)\",\n          width: \"100px\",\n          height: \"50px\",\n          zIndex: 1000000,\n          backgroundColor: \"#0085FF\",\n          color: \"white\",\n          borderRadius: 4,\n        }}\n        onClick={() => setIsOpen(!isOpen)}\n      >\n        Show Feedback\n      </div>\n      <button\n        style={{\n          position: \"fixed\",\n          top: \"calc(100vh - 122px)\",\n          left: \"calc(100vw - 124px)\",\n          width: \"100px\",\n          height: \"50px\",\n          zIndex: 1000000,\n          backgroundColor: \"#0085FF\",\n          color: \"white\",\n          borderRadius: 4,\n        }}\n        onClick={handleCreateFeedback}\n      >\n        New Feedback\n      </button>\n      {isOpen &&\n        feedbacks &&\n        feedbacks\n          .filter((feedback) => feedback.path === window.location.pathname)\n          .map((feedback) => (\n            <FeedbackItem key={feedback.id} feedback={feedback} />\n          ))}\n    </>\n  );\n}\n\nexport default Root;\n","import { QueryClient, QueryClientProvider } from \"@tanstack/react-query\";\nimport Root from \"./pages/Root\";\nimport \"./index.css\";\nimport React from \"react\";\n\nconst queryClient = new QueryClient();\n\nfunction App() {\n  return (\n    <QueryClientProvider client={queryClient}>\n      <Root />\n    </QueryClientProvider>\n  );\n}\n\nexport default App;\n"],"names":["_jsxs","_jsx","_Fragment"],"mappings":"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"}
|
|
194
|
+
//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"file":"index.es.js","sources":["../src/services/api.ts","../src/services/feedback.ts","../src/components/feedback/FeedbackItem.tsx","../src/pages/Root.tsx","../src/App.tsx"],"sourcesContent":["import axios from \"axios\";\n\nconst api = axios.create({\n  baseURL: \"http://localhost:3000\",\n});\n\napi.interceptors.request.use(\n  async (config) => {\n    // const token = localStorage.getItem(\"dashboardGaladrimAuthToken\");\n    // if (token && config.headers) {\n    //   config.headers[\"x-dashboard-token\"] = token;\n    // } else {\n    //   delete api.defaults.headers.common.Authorization;\n    // }\n    return config;\n  },\n  (error) => {\n    return Promise.reject(error);\n  }\n);\n\nexport default api;\n","import { useMutation, useQuery, useQueryClient } from \"@tanstack/react-query\";\nimport api from \"./api\";\nimport { Feedback, FeedbackPayload } from \"../types/types\";\n\nexport const fetchFeedbacks = async () => {\n  const response = await api.get<Feedback[]>(\"/feedback\");\n  return response.data;\n};\n\nexport const createFeedback = async (feedback: FeedbackPayload) => {\n  const response = await api.post(\"/feedback\", feedback);\n  return response.data;\n};\n\nexport const updateFeedback = async (feedback: FeedbackPayload) => {\n  const response = await api.put(\"/feedback\", feedback);\n  return response.data;\n};\n\nexport const deleteFeedback = async (feedback: Feedback) => {\n  const response = await api.delete(`/feedback/${feedback.id}`);\n  return response.data;\n};\n\nexport const useFeedbacks = () => {\n  return useQuery({\n    queryKey: [\"feedbacks\"],\n    queryFn: fetchFeedbacks,\n  });\n};\n\nexport const useCreateFeedback = () => {\n  const queryClient = useQueryClient();\n  return useMutation({\n    mutationFn: createFeedback,\n    onSuccess: () => {\n      console.log(\"Feedback created\");\n\n      return queryClient.invalidateQueries({ queryKey: [\"feedbacks\"] });\n    },\n  });\n};\n\nexport const useUpdateFeedback = () => {\n  const queryClient = useQueryClient();\n  return useMutation({\n    mutationFn: updateFeedback,\n    onSuccess: () => {\n      return queryClient.invalidateQueries({ queryKey: [\"feedbacks\"] });\n    },\n  });\n};\n\nexport const useDeleteFeedback = () => {\n  const queryClient = useQueryClient();\n  return useMutation({\n    mutationFn: deleteFeedback,\n    onSuccess: () => {\n      return queryClient.invalidateQueries({ queryKey: [\"feedbacks\"] });\n    },\n  });\n};\n","import { createPortal } from \"react-dom\";\nimport { useEffect, useRef, useState } from \"react\";\nimport { Feedback } from \"../../types/types\";\nimport { useDeleteFeedback } from \"../../services/feedback\";\n\nexport const FeedbackItem = ({ feedback }: { feedback: Feedback }) => {\n  const targetElement = useRef<HTMLElement | null>(null);\n  const { mutate: deleteFeedback } = useDeleteFeedback();\n  const [position, setPosition] = useState({ top: 0, left: 0 });\n\n  useEffect(() => {\n    const handleResize = () => {\n      if (!targetElement || !targetElement.current) {\n        return;\n      }\n      const dimensions = targetElement.current.getBoundingClientRect();\n      setPosition({\n        top: feedback.y + dimensions.top + window.scrollY,\n        left: feedback.x + dimensions.left + window.scrollX,\n      });\n    };\n\n    const elements = document.querySelectorAll<HTMLElement>(\n      `[data-id=\"${feedback.elementId}\"]`\n    );\n    const element = elements[feedback.elementIndex];\n\n    if (element) {\n      console.log(element);\n\n      targetElement.current = element;\n      handleResize();\n      window.addEventListener(\"resize\", handleResize);\n    }\n\n    return () => {\n      if (element) {\n        window.removeEventListener(\"resize\", handleResize);\n      }\n    };\n  }, [feedback]);\n\n  const handleDeleteFeedback = () => {\n    deleteFeedback(feedback);\n  };\n\n  // If we haven't found a matching DOM node, don't render anything\n  if (!targetElement || !targetElement.current) {\n    return null;\n  }\n\n  console.log(\"rendrr\", position);\n\n  // Render this component inside `targetElement` using a Portal\n  return createPortal(\n    <div\n      style={{\n        position: \"absolute\",\n        top: position.top,\n        left: position.left,\n        width: \"200px\",\n        height: \"100px\",\n        border: \"1px solid black\",\n        backgroundColor: \"white\",\n      }}\n    >\n      <div\n        style={{\n          display: \"flex\",\n          justifyContent: \"space-between\",\n          alignItems: \"center\",\n        }}\n      >\n        <div>\n          <p style={{ fontSize: \"16px\", fontWeight: \"bold\" }}>Feedback</p>\n          <p style={{ fontSize: \"14px\" }}>\n            Created at: {feedback.createdAt.toLocaleString()}\n          </p>\n          <a href={feedback.elementId} target=\"_blank\">\n            Go to component\n          </a>\n        </div>\n        <button onClick={handleDeleteFeedback}>Delete</button>\n      </div>\n      <p style={{ fontSize: \"14px\" }}>{feedback.comment}</p>\n    </div>,\n    document.body\n  );\n};\n","import React, { useEffect } from \"react\";\nimport { FeedbackItem } from \"../components/feedback/FeedbackItem\";\nimport { useCreateFeedback, useFeedbacks } from \"../services/feedback\";\nimport { FeedbackPayload } from \"../types/types\";\n\nfunction Root() {\n  const [isOpen, setIsOpen] = React.useState(false);\n  const { data: feedbacks } = useFeedbacks();\n\n  console.log(\"feedbacks 1 : \", feedbacks);\n\n  const { mutate: createFeedback } = useCreateFeedback();\n\n  useEffect(() => {\n    if (isOpen) {\n      window.addEventListener(\"keydown\", handleKeyPress);\n      return () => {\n        window.removeEventListener(\"keypress\", handleKeyPress);\n      };\n    }\n  }, [isOpen]);\n\n  const handleKeyPress = (event: KeyboardEvent) => {\n    if (event.key === \"Escape\") {\n      setIsOpen(false);\n    }\n  };\n\n  const handleClick = (e: MouseEvent) => {\n    e.stopPropagation();\n    e.preventDefault();\n    const node = e.target as HTMLElement | null;\n    const elementId = node?.dataset.id || \"\";\n    const allElements = document.querySelectorAll<HTMLElement>(\n      `[data-id=\"${elementId}\"]`\n    );\n\n    const arrAllElements = Array.from(allElements);\n\n    const elementIndex = arrAllElements.indexOf(node as HTMLElement);\n\n    const x = e.offsetX;\n    const y = e.offsetY;\n    console.log(e);\n    const newFeedback: FeedbackPayload = {\n      x,\n      y,\n      comment: \"test\",\n      path: window.location.pathname,\n      elementId,\n      elementIndex,\n    };\n    console.log(newFeedback);\n    createFeedback(newFeedback);\n    document.body.classList.remove(\"comment-cursor\");\n    document.removeEventListener(\"click\", handleClick, true);\n    setIsOpen(true);\n  };\n\n  const handleCreateFeedback = (e: React.MouseEvent<HTMLButtonElement>) => {\n    e.stopPropagation();\n    document.body.classList.add(\"comment-cursor\");\n    document.addEventListener(\"click\", handleClick, true);\n  };\n\n  return (\n    <>\n      <div\n        style={{\n          position: \"fixed\",\n          top: \"calc(100vh - 62px)\",\n          left: \"calc(100vw - 124px)\",\n          width: \"100px\",\n          height: \"50px\",\n          zIndex: 1000000,\n          backgroundColor: \"#0085FF\",\n          color: \"white\",\n          borderRadius: 4,\n        }}\n        onClick={() => setIsOpen(!isOpen)}\n      >\n        Show Feedback\n      </div>\n      <button\n        style={{\n          position: \"fixed\",\n          top: \"calc(100vh - 122px)\",\n          left: \"calc(100vw - 124px)\",\n          width: \"100px\",\n          height: \"50px\",\n          zIndex: 1000000,\n          backgroundColor: \"#0085FF\",\n          color: \"white\",\n          borderRadius: 4,\n        }}\n        onClick={handleCreateFeedback}\n      >\n        New Feedback\n      </button>\n      {isOpen &&\n        feedbacks &&\n        feedbacks\n          .filter((feedback) => feedback.path === window.location.pathname)\n          .map((feedback) => (\n            <FeedbackItem key={feedback.id} feedback={feedback} />\n          ))}\n    </>\n  );\n}\n\nexport default Root;\n","import { QueryClient, QueryClientProvider } from \"@tanstack/react-query\";\nimport Root from \"./pages/Root\";\nimport \"./index.css\";\nimport React from \"react\";\n\nconst queryClient = new QueryClient();\n\nfunction App() {\n  return (\n    <QueryClientProvider client={queryClient}>\n      <Root />\n    </QueryClientProvider>\n  );\n}\n\nexport default App;\n"],"names":["_jsxs","_jsx","_Fragment"],"mappings":"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"}
|
package/dist/src/index.css
CHANGED
package/package.json
CHANGED
|
@@ -1,11 +1,15 @@
|
|
|
1
1
|
{
|
|
2
2
|
"name": "galadrim-feedback",
|
|
3
3
|
"private": false,
|
|
4
|
-
"version": "0.0.
|
|
4
|
+
"version": "0.0.4",
|
|
5
5
|
"type": "module",
|
|
6
6
|
"main": "dist/index.cjs.js",
|
|
7
7
|
"module": "dist/index.es.js",
|
|
8
8
|
"types": "dist/index.d.ts",
|
|
9
|
+
"files": [
|
|
10
|
+
"dist/",
|
|
11
|
+
"assets/"
|
|
12
|
+
],
|
|
9
13
|
"scripts": {
|
|
10
14
|
"build": "rollup -c",
|
|
11
15
|
"prepublishOnly": "npm run build"
|
|
@@ -25,6 +29,7 @@
|
|
|
25
29
|
"@rollup/plugin-json": "^6.1.0",
|
|
26
30
|
"@rollup/plugin-node-resolve": "^16.0.0",
|
|
27
31
|
"@rollup/plugin-typescript": "^12.1.2",
|
|
32
|
+
"@rollup/plugin-url": "^8.0.2",
|
|
28
33
|
"@types/react": "^18.3.10",
|
|
29
34
|
"@types/react-dom": "^18.3.0",
|
|
30
35
|
"@vitejs/plugin-react": "^4.3.2",
|
|
@@ -36,6 +41,7 @@
|
|
|
36
41
|
"globals": "^15.9.0",
|
|
37
42
|
"postcss": "^8.4.49",
|
|
38
43
|
"rollup": "^4.30.1",
|
|
44
|
+
"rollup-plugin-copy": "^3.5.0",
|
|
39
45
|
"rollup-plugin-node-polyfills": "^0.2.1",
|
|
40
46
|
"rollup-plugin-postcss": "^4.0.2",
|
|
41
47
|
"sass": "^1.83.1",
|
package/eslint.config.js
DELETED
|
@@ -1,28 +0,0 @@
|
|
|
1
|
-
import js from '@eslint/js'
|
|
2
|
-
import globals from 'globals'
|
|
3
|
-
import reactHooks from 'eslint-plugin-react-hooks'
|
|
4
|
-
import reactRefresh from 'eslint-plugin-react-refresh'
|
|
5
|
-
import tseslint from 'typescript-eslint'
|
|
6
|
-
|
|
7
|
-
export default tseslint.config(
|
|
8
|
-
{ ignores: ['dist'] },
|
|
9
|
-
{
|
|
10
|
-
extends: [js.configs.recommended, ...tseslint.configs.recommended],
|
|
11
|
-
files: ['**/*.{ts,tsx}'],
|
|
12
|
-
languageOptions: {
|
|
13
|
-
ecmaVersion: 2020,
|
|
14
|
-
globals: globals.browser,
|
|
15
|
-
},
|
|
16
|
-
plugins: {
|
|
17
|
-
'react-hooks': reactHooks,
|
|
18
|
-
'react-refresh': reactRefresh,
|
|
19
|
-
},
|
|
20
|
-
rules: {
|
|
21
|
-
...reactHooks.configs.recommended.rules,
|
|
22
|
-
'react-refresh/only-export-components': [
|
|
23
|
-
'warn',
|
|
24
|
-
{ allowConstantExport: true },
|
|
25
|
-
],
|
|
26
|
-
},
|
|
27
|
-
},
|
|
28
|
-
)
|
package/rollup.config.js
DELETED
|
@@ -1,40 +0,0 @@
|
|
|
1
|
-
// rollup.config.js
|
|
2
|
-
import commonjs from "@rollup/plugin-commonjs";
|
|
3
|
-
import resolve from "@rollup/plugin-node-resolve";
|
|
4
|
-
import typescript from "@rollup/plugin-typescript";
|
|
5
|
-
import postcss from "rollup-plugin-postcss";
|
|
6
|
-
import json from "@rollup/plugin-json";
|
|
7
|
-
import nodePolyfills from "rollup-plugin-node-polyfills";
|
|
8
|
-
import pkg from "./package.json" assert { type: "json" };
|
|
9
|
-
|
|
10
|
-
export default {
|
|
11
|
-
input: "src/App.tsx", // Your entry point
|
|
12
|
-
output: [
|
|
13
|
-
{
|
|
14
|
-
file: pkg.main, // dist/index.cjs.js
|
|
15
|
-
format: "cjs",
|
|
16
|
-
sourcemap: "inline",
|
|
17
|
-
},
|
|
18
|
-
{
|
|
19
|
-
file: pkg.module, // dist/index.es.js
|
|
20
|
-
format: "es",
|
|
21
|
-
sourcemap: "inline",
|
|
22
|
-
},
|
|
23
|
-
],
|
|
24
|
-
external: [
|
|
25
|
-
// Peer dependencies (don’t bundle React, ReactDOM, etc.)
|
|
26
|
-
...Object.keys(pkg.peerDependencies || {}),
|
|
27
|
-
],
|
|
28
|
-
plugins: [
|
|
29
|
-
resolve(),
|
|
30
|
-
nodePolyfills(),
|
|
31
|
-
commonjs(),
|
|
32
|
-
postcss({
|
|
33
|
-
extract: "src/index.css",
|
|
34
|
-
}),
|
|
35
|
-
typescript({
|
|
36
|
-
tsconfig: "./tsconfig.json",
|
|
37
|
-
}),
|
|
38
|
-
json(),
|
|
39
|
-
],
|
|
40
|
-
};
|
package/src/App.tsx
DELETED
|
@@ -1,16 +0,0 @@
|
|
|
1
|
-
import { QueryClient, QueryClientProvider } from "@tanstack/react-query";
|
|
2
|
-
import Root from "./pages/Root";
|
|
3
|
-
import "./index.css";
|
|
4
|
-
import React from "react";
|
|
5
|
-
|
|
6
|
-
const queryClient = new QueryClient();
|
|
7
|
-
|
|
8
|
-
function App() {
|
|
9
|
-
return (
|
|
10
|
-
<QueryClientProvider client={queryClient}>
|
|
11
|
-
<Root />
|
|
12
|
-
</QueryClientProvider>
|
|
13
|
-
);
|
|
14
|
-
}
|
|
15
|
-
|
|
16
|
-
export default App;
|
|
@@ -1,27 +0,0 @@
|
|
|
1
|
-
import classNames from "classnames";
|
|
2
|
-
|
|
3
|
-
export const IconButton = ({
|
|
4
|
-
text,
|
|
5
|
-
iconPath,
|
|
6
|
-
twButtonBackgroundColor,
|
|
7
|
-
onClick,
|
|
8
|
-
}: {
|
|
9
|
-
text: string;
|
|
10
|
-
iconPath: string;
|
|
11
|
-
twButtonBackgroundColor: string;
|
|
12
|
-
onClick: () => void;
|
|
13
|
-
}) => {
|
|
14
|
-
return (
|
|
15
|
-
<button
|
|
16
|
-
onClick={onClick}
|
|
17
|
-
className={classNames(
|
|
18
|
-
"flex-shrink-0 flex flex-row items-center px-4 py-2 gap-2 rounded-[8px] text-white font-[500] ",
|
|
19
|
-
twButtonBackgroundColor,
|
|
20
|
-
"hover:bg-opacity-80"
|
|
21
|
-
)}
|
|
22
|
-
>
|
|
23
|
-
<img src={iconPath} className="w-6 h-6" />
|
|
24
|
-
{text}
|
|
25
|
-
</button>
|
|
26
|
-
);
|
|
27
|
-
};
|
|
@@ -1,20 +0,0 @@
|
|
|
1
|
-
import { IconButton } from "./IconButton";
|
|
2
|
-
|
|
3
|
-
export const PlusButton = ({
|
|
4
|
-
text,
|
|
5
|
-
twButtonBackgroundColor,
|
|
6
|
-
onClick,
|
|
7
|
-
}: {
|
|
8
|
-
text: string;
|
|
9
|
-
twButtonBackgroundColor: string;
|
|
10
|
-
onClick: () => void;
|
|
11
|
-
}) => {
|
|
12
|
-
return (
|
|
13
|
-
<IconButton
|
|
14
|
-
text={text}
|
|
15
|
-
iconPath="/plus.svg"
|
|
16
|
-
twButtonBackgroundColor={twButtonBackgroundColor}
|
|
17
|
-
onClick={onClick}
|
|
18
|
-
/>
|
|
19
|
-
);
|
|
20
|
-
};
|
|
@@ -1,89 +0,0 @@
|
|
|
1
|
-
import { createPortal } from "react-dom";
|
|
2
|
-
import { useEffect, useRef, useState } from "react";
|
|
3
|
-
import { Feedback } from "../../types/types";
|
|
4
|
-
import { useDeleteFeedback } from "../../services/feedback";
|
|
5
|
-
|
|
6
|
-
export const FeedbackItem = ({ feedback }: { feedback: Feedback }) => {
|
|
7
|
-
const targetElement = useRef<HTMLElement | null>(null);
|
|
8
|
-
const { mutate: deleteFeedback } = useDeleteFeedback();
|
|
9
|
-
const [position, setPosition] = useState({ top: 0, left: 0 });
|
|
10
|
-
|
|
11
|
-
useEffect(() => {
|
|
12
|
-
const handleResize = () => {
|
|
13
|
-
if (!targetElement || !targetElement.current) {
|
|
14
|
-
return;
|
|
15
|
-
}
|
|
16
|
-
const dimensions = targetElement.current.getBoundingClientRect();
|
|
17
|
-
setPosition({
|
|
18
|
-
top: feedback.y + dimensions.top + window.scrollY,
|
|
19
|
-
left: feedback.x + dimensions.left + window.scrollX,
|
|
20
|
-
});
|
|
21
|
-
};
|
|
22
|
-
|
|
23
|
-
const elements = document.querySelectorAll<HTMLElement>(
|
|
24
|
-
`[data-id="${feedback.elementId}"]`
|
|
25
|
-
);
|
|
26
|
-
const element = elements[feedback.elementIndex];
|
|
27
|
-
|
|
28
|
-
if (element) {
|
|
29
|
-
console.log(element);
|
|
30
|
-
|
|
31
|
-
targetElement.current = element;
|
|
32
|
-
handleResize();
|
|
33
|
-
window.addEventListener("resize", handleResize);
|
|
34
|
-
}
|
|
35
|
-
|
|
36
|
-
return () => {
|
|
37
|
-
if (element) {
|
|
38
|
-
window.removeEventListener("resize", handleResize);
|
|
39
|
-
}
|
|
40
|
-
};
|
|
41
|
-
}, [feedback]);
|
|
42
|
-
|
|
43
|
-
const handleDeleteFeedback = () => {
|
|
44
|
-
deleteFeedback(feedback);
|
|
45
|
-
};
|
|
46
|
-
|
|
47
|
-
// If we haven't found a matching DOM node, don't render anything
|
|
48
|
-
if (!targetElement || !targetElement.current) {
|
|
49
|
-
return null;
|
|
50
|
-
}
|
|
51
|
-
|
|
52
|
-
console.log("rendrr", position);
|
|
53
|
-
|
|
54
|
-
// Render this component inside `targetElement` using a Portal
|
|
55
|
-
return createPortal(
|
|
56
|
-
<div
|
|
57
|
-
style={{
|
|
58
|
-
position: "absolute",
|
|
59
|
-
top: position.top,
|
|
60
|
-
left: position.left,
|
|
61
|
-
width: "200px",
|
|
62
|
-
height: "100px",
|
|
63
|
-
border: "1px solid black",
|
|
64
|
-
backgroundColor: "white",
|
|
65
|
-
}}
|
|
66
|
-
>
|
|
67
|
-
<div
|
|
68
|
-
style={{
|
|
69
|
-
display: "flex",
|
|
70
|
-
justifyContent: "space-between",
|
|
71
|
-
alignItems: "center",
|
|
72
|
-
}}
|
|
73
|
-
>
|
|
74
|
-
<div>
|
|
75
|
-
<p style={{ fontSize: "16px", fontWeight: "bold" }}>Feedback</p>
|
|
76
|
-
<p style={{ fontSize: "14px" }}>
|
|
77
|
-
Created at: {feedback.createdAt.toLocaleString()}
|
|
78
|
-
</p>
|
|
79
|
-
<a href={feedback.elementId} target="_blank">
|
|
80
|
-
Go to component
|
|
81
|
-
</a>
|
|
82
|
-
</div>
|
|
83
|
-
<button onClick={handleDeleteFeedback}>Delete</button>
|
|
84
|
-
</div>
|
|
85
|
-
<p style={{ fontSize: "14px" }}>{feedback.comment}</p>
|
|
86
|
-
</div>,
|
|
87
|
-
document.body
|
|
88
|
-
);
|
|
89
|
-
};
|
|
@@ -1,43 +0,0 @@
|
|
|
1
|
-
import { useRef } from "react";
|
|
2
|
-
|
|
3
|
-
export const Modal = ({
|
|
4
|
-
isVisible,
|
|
5
|
-
closeModal,
|
|
6
|
-
limitWidth = true,
|
|
7
|
-
children,
|
|
8
|
-
}: {
|
|
9
|
-
isVisible: boolean;
|
|
10
|
-
closeModal?: () => void;
|
|
11
|
-
limitWidth?: boolean;
|
|
12
|
-
children: React.ReactNode;
|
|
13
|
-
}) => {
|
|
14
|
-
const resizing = useRef(false);
|
|
15
|
-
|
|
16
|
-
if (!isVisible) {
|
|
17
|
-
return null;
|
|
18
|
-
}
|
|
19
|
-
const width = limitWidth ? "w-[400px]" : "";
|
|
20
|
-
return (
|
|
21
|
-
<div
|
|
22
|
-
className="modal"
|
|
23
|
-
onMouseDown={(e) => {
|
|
24
|
-
const target = e.target as HTMLElement;
|
|
25
|
-
if (target.tagName === "TEXTAREA") {
|
|
26
|
-
resizing.current = true;
|
|
27
|
-
}
|
|
28
|
-
}}
|
|
29
|
-
onMouseUp={() => {
|
|
30
|
-
setTimeout(() => {
|
|
31
|
-
resizing.current = false;
|
|
32
|
-
}, 100);
|
|
33
|
-
}}
|
|
34
|
-
onClick={(e) => {
|
|
35
|
-
if (closeModal && e.target === e.currentTarget && !resizing.current) {
|
|
36
|
-
closeModal();
|
|
37
|
-
}
|
|
38
|
-
}}
|
|
39
|
-
>
|
|
40
|
-
<div className={`${width} bg-white rounded-lg`}>{children}</div>
|
|
41
|
-
</div>
|
|
42
|
-
);
|
|
43
|
-
};
|
package/src/index.css
DELETED
|
@@ -1,38 +0,0 @@
|
|
|
1
|
-
import { Dispatch, useEffect } from "react";
|
|
2
|
-
import { FeedbackItem } from "../components/feedback/FeedbackItem";
|
|
3
|
-
import { useFeedbacks } from "../services/feedback";
|
|
4
|
-
import { Feedback } from "../types/types";
|
|
5
|
-
|
|
6
|
-
export const FeedbacksCanvas = ({
|
|
7
|
-
isOpen,
|
|
8
|
-
setIsOpen,
|
|
9
|
-
feedbacks,
|
|
10
|
-
}: {
|
|
11
|
-
isOpen: boolean;
|
|
12
|
-
setIsOpen: Dispatch<React.SetStateAction<boolean>>;
|
|
13
|
-
feedbacks: Feedback[];
|
|
14
|
-
}) => {
|
|
15
|
-
if (!isOpen) {
|
|
16
|
-
return null;
|
|
17
|
-
}
|
|
18
|
-
|
|
19
|
-
return (
|
|
20
|
-
<div
|
|
21
|
-
style={{
|
|
22
|
-
zIndex: 2147483647, // Maximum zIndex
|
|
23
|
-
position: "fixed",
|
|
24
|
-
top: 0,
|
|
25
|
-
left: 0,
|
|
26
|
-
width: "100vw",
|
|
27
|
-
height: "100vh",
|
|
28
|
-
}}
|
|
29
|
-
>
|
|
30
|
-
{feedbacks &&
|
|
31
|
-
feedbacks
|
|
32
|
-
.filter((feedback) => feedback.path === window.location.pathname)
|
|
33
|
-
.map((feedback) => (
|
|
34
|
-
<FeedbackItem key={feedback.id} feedback={feedback} />
|
|
35
|
-
))}
|
|
36
|
-
</div>
|
|
37
|
-
);
|
|
38
|
-
};
|
package/src/pages/Overlay.tsx
DELETED
|
@@ -1,26 +0,0 @@
|
|
|
1
|
-
import { QueryClient, QueryClientProvider } from "@tanstack/react-query";
|
|
2
|
-
import { Dispatch } from "react";
|
|
3
|
-
import { FeedbacksCanvas } from "./FeedbacksCanvas";
|
|
4
|
-
import { Feedback } from "../types/types";
|
|
5
|
-
|
|
6
|
-
export const Overlay = ({
|
|
7
|
-
isOpen,
|
|
8
|
-
setIsOpen,
|
|
9
|
-
queryClient,
|
|
10
|
-
feedbacks,
|
|
11
|
-
}: {
|
|
12
|
-
isOpen: boolean;
|
|
13
|
-
setIsOpen: Dispatch<React.SetStateAction<boolean>>;
|
|
14
|
-
queryClient: QueryClient;
|
|
15
|
-
feedbacks: Feedback[];
|
|
16
|
-
}) => {
|
|
17
|
-
return (
|
|
18
|
-
<QueryClientProvider client={queryClient}>
|
|
19
|
-
<FeedbacksCanvas
|
|
20
|
-
isOpen={isOpen}
|
|
21
|
-
setIsOpen={setIsOpen}
|
|
22
|
-
feedbacks={feedbacks}
|
|
23
|
-
/>
|
|
24
|
-
</QueryClientProvider>
|
|
25
|
-
);
|
|
26
|
-
};
|
package/src/pages/Root.tsx
DELETED
|
@@ -1,128 +0,0 @@
|
|
|
1
|
-
import { QueryClient } from "@tanstack/react-query";
|
|
2
|
-
import React, { useEffect } from "react";
|
|
3
|
-
import ReactDOM from "react-dom";
|
|
4
|
-
import { Overlay } from "./Overlay";
|
|
5
|
-
import { useCreateFeedback, useFeedbacks } from "../services/feedback";
|
|
6
|
-
import { FeedbackPayload } from "../types/types";
|
|
7
|
-
import { FeedbackItem } from "../components/feedback/FeedbackItem";
|
|
8
|
-
|
|
9
|
-
const queryClient = new QueryClient();
|
|
10
|
-
|
|
11
|
-
function Root() {
|
|
12
|
-
const [isOpen, setIsOpen] = React.useState(false);
|
|
13
|
-
const { data: feedbacks } = useFeedbacks();
|
|
14
|
-
|
|
15
|
-
console.log("feedbacks 1 : ", feedbacks);
|
|
16
|
-
|
|
17
|
-
const [creatingFeedback, setCreatingFeedback] = React.useState(false);
|
|
18
|
-
|
|
19
|
-
const { mutate: createFeedback } = useCreateFeedback();
|
|
20
|
-
|
|
21
|
-
useEffect(() => {
|
|
22
|
-
if (isOpen) {
|
|
23
|
-
console.log("Opening feedbacks canvas");
|
|
24
|
-
|
|
25
|
-
window.addEventListener("keydown", handleKeyPress);
|
|
26
|
-
return () => {
|
|
27
|
-
window.removeEventListener("keypress", handleKeyPress);
|
|
28
|
-
};
|
|
29
|
-
}
|
|
30
|
-
}, [isOpen]);
|
|
31
|
-
|
|
32
|
-
const handleKeyPress = (event: KeyboardEvent) => {
|
|
33
|
-
console.log(event.key);
|
|
34
|
-
|
|
35
|
-
if (event.key === "Escape") {
|
|
36
|
-
setIsOpen(false);
|
|
37
|
-
}
|
|
38
|
-
};
|
|
39
|
-
|
|
40
|
-
const handleClick = (e: any) => {
|
|
41
|
-
console.log("click");
|
|
42
|
-
|
|
43
|
-
e.stopPropagation();
|
|
44
|
-
e.preventDefault();
|
|
45
|
-
const node = e.target as HTMLElement | null;
|
|
46
|
-
const elementId = node?.dataset.id || "";
|
|
47
|
-
const allElements = document.querySelectorAll<HTMLElement>(
|
|
48
|
-
`[data-id="${elementId}"]`
|
|
49
|
-
);
|
|
50
|
-
|
|
51
|
-
const arrAllElements = Array.from(allElements);
|
|
52
|
-
|
|
53
|
-
const elementIndex = arrAllElements.indexOf(node as HTMLElement);
|
|
54
|
-
|
|
55
|
-
const x = e.offsetX;
|
|
56
|
-
const y = e.offsetY;
|
|
57
|
-
console.log(e);
|
|
58
|
-
const newFeedback: FeedbackPayload = {
|
|
59
|
-
x,
|
|
60
|
-
y,
|
|
61
|
-
screenWidth: window.innerWidth,
|
|
62
|
-
screenHeight: window.innerHeight,
|
|
63
|
-
comment: "test",
|
|
64
|
-
path: window.location.pathname,
|
|
65
|
-
elementId,
|
|
66
|
-
elementIndex,
|
|
67
|
-
};
|
|
68
|
-
console.log(newFeedback);
|
|
69
|
-
createFeedback(newFeedback);
|
|
70
|
-
document.body.classList.remove("comment-cursor");
|
|
71
|
-
document.removeEventListener("click", handleClick, true);
|
|
72
|
-
setCreatingFeedback(false);
|
|
73
|
-
setIsOpen(true);
|
|
74
|
-
};
|
|
75
|
-
|
|
76
|
-
const handleCreateFeedback = (e: React.MouseEvent<HTMLButtonElement>) => {
|
|
77
|
-
e.stopPropagation();
|
|
78
|
-
setCreatingFeedback(true);
|
|
79
|
-
document.body.classList.add("comment-cursor");
|
|
80
|
-
document.addEventListener("click", handleClick, true);
|
|
81
|
-
};
|
|
82
|
-
|
|
83
|
-
return (
|
|
84
|
-
<>
|
|
85
|
-
<div
|
|
86
|
-
style={{
|
|
87
|
-
position: "fixed",
|
|
88
|
-
top: "calc(100vh - 62px)",
|
|
89
|
-
left: "calc(100vw - 124px)",
|
|
90
|
-
width: "100px",
|
|
91
|
-
height: "50px",
|
|
92
|
-
zIndex: 1000000,
|
|
93
|
-
backgroundColor: "#0085FF",
|
|
94
|
-
color: "white",
|
|
95
|
-
borderRadius: 4,
|
|
96
|
-
}}
|
|
97
|
-
onClick={() => setIsOpen(!isOpen)}
|
|
98
|
-
>
|
|
99
|
-
Show Feedback
|
|
100
|
-
</div>
|
|
101
|
-
<button
|
|
102
|
-
style={{
|
|
103
|
-
position: "fixed",
|
|
104
|
-
top: "calc(100vh - 122px)",
|
|
105
|
-
left: "calc(100vw - 124px)",
|
|
106
|
-
width: "100px",
|
|
107
|
-
height: "50px",
|
|
108
|
-
zIndex: 1000000,
|
|
109
|
-
backgroundColor: "#0085FF",
|
|
110
|
-
color: "white",
|
|
111
|
-
borderRadius: 4,
|
|
112
|
-
}}
|
|
113
|
-
onClick={handleCreateFeedback}
|
|
114
|
-
>
|
|
115
|
-
New Feedback
|
|
116
|
-
</button>
|
|
117
|
-
{isOpen &&
|
|
118
|
-
feedbacks &&
|
|
119
|
-
feedbacks
|
|
120
|
-
.filter((feedback) => feedback.path === window.location.pathname)
|
|
121
|
-
.map((feedback) => (
|
|
122
|
-
<FeedbackItem key={feedback.id} feedback={feedback} />
|
|
123
|
-
))}
|
|
124
|
-
</>
|
|
125
|
-
);
|
|
126
|
-
}
|
|
127
|
-
|
|
128
|
-
export default Root;
|
package/src/services/api.ts
DELETED
|
@@ -1,22 +0,0 @@
|
|
|
1
|
-
import axios from "axios";
|
|
2
|
-
|
|
3
|
-
const api = axios.create({
|
|
4
|
-
baseURL: "http://localhost:3000",
|
|
5
|
-
});
|
|
6
|
-
|
|
7
|
-
api.interceptors.request.use(
|
|
8
|
-
async (config) => {
|
|
9
|
-
// const token = localStorage.getItem("dashboardGaladrimAuthToken");
|
|
10
|
-
// if (token && config.headers) {
|
|
11
|
-
// config.headers["x-dashboard-token"] = token;
|
|
12
|
-
// } else {
|
|
13
|
-
// delete api.defaults.headers.common.Authorization;
|
|
14
|
-
// }
|
|
15
|
-
return config;
|
|
16
|
-
},
|
|
17
|
-
(error) => {
|
|
18
|
-
return Promise.reject(error);
|
|
19
|
-
}
|
|
20
|
-
);
|
|
21
|
-
|
|
22
|
-
export default api;
|
package/src/services/feedback.ts
DELETED
|
@@ -1,62 +0,0 @@
|
|
|
1
|
-
import { useMutation, useQuery, useQueryClient } from "@tanstack/react-query";
|
|
2
|
-
import api from "./api";
|
|
3
|
-
import { Feedback, FeedbackPayload } from "../types/types";
|
|
4
|
-
|
|
5
|
-
export const fetchFeedbacks = async () => {
|
|
6
|
-
const response = await api.get<Feedback[]>("/feedback");
|
|
7
|
-
return response.data;
|
|
8
|
-
};
|
|
9
|
-
|
|
10
|
-
export const createFeedback = async (feedback: FeedbackPayload) => {
|
|
11
|
-
const response = await api.post("/feedback", feedback);
|
|
12
|
-
return response.data;
|
|
13
|
-
};
|
|
14
|
-
|
|
15
|
-
export const updateFeedback = async (feedback: FeedbackPayload) => {
|
|
16
|
-
const response = await api.put("/feedback", feedback);
|
|
17
|
-
return response.data;
|
|
18
|
-
};
|
|
19
|
-
|
|
20
|
-
export const deleteFeedback = async (feedback: Feedback) => {
|
|
21
|
-
const response = await api.delete(`/feedback/${feedback.id}`);
|
|
22
|
-
return response.data;
|
|
23
|
-
};
|
|
24
|
-
|
|
25
|
-
export const useFeedbacks = () => {
|
|
26
|
-
return useQuery({
|
|
27
|
-
queryKey: ["feedbacks"],
|
|
28
|
-
queryFn: fetchFeedbacks,
|
|
29
|
-
});
|
|
30
|
-
};
|
|
31
|
-
|
|
32
|
-
export const useCreateFeedback = () => {
|
|
33
|
-
const queryClient = useQueryClient();
|
|
34
|
-
return useMutation({
|
|
35
|
-
mutationFn: createFeedback,
|
|
36
|
-
onSuccess: () => {
|
|
37
|
-
console.log("Feedback created");
|
|
38
|
-
|
|
39
|
-
return queryClient.invalidateQueries({ queryKey: ["feedbacks"] });
|
|
40
|
-
},
|
|
41
|
-
});
|
|
42
|
-
};
|
|
43
|
-
|
|
44
|
-
export const useUpdateFeedback = () => {
|
|
45
|
-
const queryClient = useQueryClient();
|
|
46
|
-
return useMutation({
|
|
47
|
-
mutationFn: updateFeedback,
|
|
48
|
-
onSuccess: () => {
|
|
49
|
-
return queryClient.invalidateQueries({ queryKey: ["feedbacks"] });
|
|
50
|
-
},
|
|
51
|
-
});
|
|
52
|
-
};
|
|
53
|
-
|
|
54
|
-
export const useDeleteFeedback = () => {
|
|
55
|
-
const queryClient = useQueryClient();
|
|
56
|
-
return useMutation({
|
|
57
|
-
mutationFn: deleteFeedback,
|
|
58
|
-
onSuccess: () => {
|
|
59
|
-
return queryClient.invalidateQueries({ queryKey: ["feedbacks"] });
|
|
60
|
-
},
|
|
61
|
-
});
|
|
62
|
-
};
|
package/src/types/types.d.ts
DELETED
|
@@ -1,16 +0,0 @@
|
|
|
1
|
-
export interface FeedbackPayload {
|
|
2
|
-
x: number;
|
|
3
|
-
y: number;
|
|
4
|
-
screenWidth: number;
|
|
5
|
-
screenHeight: number;
|
|
6
|
-
elementId: string;
|
|
7
|
-
elementIndex: number;
|
|
8
|
-
comment: string;
|
|
9
|
-
path: string;
|
|
10
|
-
}
|
|
11
|
-
|
|
12
|
-
export interface Feedback extends FeedbackPayload {
|
|
13
|
-
id: string;
|
|
14
|
-
createdAt: Date;
|
|
15
|
-
updatedAt: Date;
|
|
16
|
-
}
|
package/tailwind.config.js
DELETED
|
@@ -1,46 +0,0 @@
|
|
|
1
|
-
/** @type {import('tailwindcss').Config} */
|
|
2
|
-
export default {
|
|
3
|
-
content: ["./index.html", "./src/**/*.{js,ts,jsx,tsx}"],
|
|
4
|
-
theme: {
|
|
5
|
-
extend: {
|
|
6
|
-
colors: {
|
|
7
|
-
gray: {
|
|
8
|
-
50: "#F2F2F2",
|
|
9
|
-
75: "#F4F5F7",
|
|
10
|
-
100: "#E2E8F0",
|
|
11
|
-
200: "#EAECF0",
|
|
12
|
-
300: "#676D78",
|
|
13
|
-
400: "#343D4B",
|
|
14
|
-
500: "#010C1E",
|
|
15
|
-
600: "#475467",
|
|
16
|
-
700: "#CBD5E1",
|
|
17
|
-
900: "#101828",
|
|
18
|
-
},
|
|
19
|
-
blue: {
|
|
20
|
-
50: "#F5FAFF",
|
|
21
|
-
100: "#EBF5FF",
|
|
22
|
-
500: "#0085FF",
|
|
23
|
-
},
|
|
24
|
-
red: {
|
|
25
|
-
100: "#FFD3E0",
|
|
26
|
-
500: "#FF2462",
|
|
27
|
-
},
|
|
28
|
-
rose: {
|
|
29
|
-
100: "#FFD3E0",
|
|
30
|
-
200: "#FFA7C0",
|
|
31
|
-
},
|
|
32
|
-
orange: {
|
|
33
|
-
50: "#FEF6EE",
|
|
34
|
-
200: "#F9DBAF",
|
|
35
|
-
700: "#B93815",
|
|
36
|
-
},
|
|
37
|
-
green: {
|
|
38
|
-
50: "#ECFDF3",
|
|
39
|
-
200: "#ABEFC6",
|
|
40
|
-
700: "#067647",
|
|
41
|
-
},
|
|
42
|
-
},
|
|
43
|
-
},
|
|
44
|
-
},
|
|
45
|
-
plugins: [],
|
|
46
|
-
};
|
package/tsconfig.json
DELETED
|
@@ -1,18 +0,0 @@
|
|
|
1
|
-
{
|
|
2
|
-
"compilerOptions": {
|
|
3
|
-
"target": "ES2017",
|
|
4
|
-
"module": "ESNext",
|
|
5
|
-
"jsx": "react-jsx",
|
|
6
|
-
// Generate type definitions
|
|
7
|
-
"declaration": true,
|
|
8
|
-
"declarationDir": "dist/types",
|
|
9
|
-
"outDir": "dist",
|
|
10
|
-
"moduleResolution": "node",
|
|
11
|
-
"isolatedModules": true,
|
|
12
|
-
"esModuleInterop": true,
|
|
13
|
-
"skipLibCheck": true,
|
|
14
|
-
"strict": true
|
|
15
|
-
},
|
|
16
|
-
"include": ["src"],
|
|
17
|
-
"exclude": ["node_modules", "**/*.test.*"]
|
|
18
|
-
}
|