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 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":";;;;;;;;AAEA,MAAM,GAAG,GAAG,KAAK,CAAC,MAAM,CAAC;AACvB,IAAA,OAAO,EAAE,uBAAuB;AACjC,CAAA,CAAC;AAEF,GAAG,CAAC,YAAY,CAAC,OAAO,CAAC,GAAG,CAC1B,OAAO,MAAM,KAAI;;;;;;;AAOf,IAAA,OAAO,MAAM;AACf,CAAC,EACD,CAAC,KAAK,KAAI;AACR,IAAA,OAAO,OAAO,CAAC,MAAM,CAAC,KAAK,CAAC;AAC9B,CAAC,CACF;;ACfM,MAAM,cAAc,GAAG,YAAW;IACvC,MAAM,QAAQ,GAAG,MAAM,GAAG,CAAC,GAAG,CAAa,WAAW,CAAC;IACvD,OAAO,QAAQ,CAAC,IAAI;AACtB,CAAC;AAEM,MAAM,cAAc,GAAG,OAAO,QAAyB,KAAI;IAChE,MAAM,QAAQ,GAAG,MAAM,GAAG,CAAC,IAAI,CAAC,WAAW,EAAE,QAAQ,CAAC;IACtD,OAAO,QAAQ,CAAC,IAAI;AACtB,CAAC;AAOM,MAAM,cAAc,GAAG,OAAO,QAAkB,KAAI;AACzD,IAAA,MAAM,QAAQ,GAAG,MAAM,GAAG,CAAC,MAAM,CAAC,CAAA,UAAA,EAAa,QAAQ,CAAC,EAAE,CAAA,CAAE,CAAC;IAC7D,OAAO,QAAQ,CAAC,IAAI;AACtB,CAAC;AAEM,MAAM,YAAY,GAAG,MAAK;AAC/B,IAAA,OAAOA,mBAAQ,CAAC;QACd,QAAQ,EAAE,CAAC,WAAW,CAAC;AACvB,QAAA,OAAO,EAAE,cAAc;AACxB,KAAA,CAAC;AACJ,CAAC;AAEM,MAAM,iBAAiB,GAAG,MAAK;AACpC,IAAA,MAAM,WAAW,GAAGC,yBAAc,EAAE;AACpC,IAAA,OAAOC,sBAAW,CAAC;AACjB,QAAA,UAAU,EAAE,cAAc;QAC1B,SAAS,EAAE,MAAK;AACd,YAAA,OAAO,CAAC,GAAG,CAAC,kBAAkB,CAAC;AAE/B,YAAA,OAAO,WAAW,CAAC,iBAAiB,CAAC,EAAE,QAAQ,EAAE,CAAC,WAAW,CAAC,EAAE,CAAC;SAClE;AACF,KAAA,CAAC;AACJ,CAAC;AAYM,MAAM,iBAAiB,GAAG,MAAK;AACpC,IAAA,MAAM,WAAW,GAAGD,yBAAc,EAAE;AACpC,IAAA,OAAOC,sBAAW,CAAC;AACjB,QAAA,UAAU,EAAE,cAAc;QAC1B,SAAS,EAAE,MAAK;AACd,YAAA,OAAO,WAAW,CAAC,iBAAiB,CAAC,EAAE,QAAQ,EAAE,CAAC,WAAW,CAAC,EAAE,CAAC;SAClE;AACF,KAAA,CAAC;AACJ,CAAC;;ACxDM,MAAM,YAAY,GAAG,CAAC,EAAE,QAAQ,EAA0B,KAAI;AACnE,IAAA,MAAM,aAAa,GAAGC,YAAM,CAAqB,IAAI,CAAC;IACtD,MAAM,EAAE,MAAM,EAAE,cAAc,EAAE,GAAG,iBAAiB,EAAE;AACtD,IAAA,MAAM,CAAC,QAAQ,EAAE,WAAW,CAAC,GAAGC,cAAQ,CAAC,EAAE,GAAG,EAAE,CAAC,EAAE,IAAI,EAAE,CAAC,EAAE,CAAC;IAE7DC,eAAS,CAAC,MAAK;QACb,MAAM,YAAY,GAAG,MAAK;YACxB,IAAI,CAAC,aAAa,IAAI,CAAC,aAAa,CAAC,OAAO,EAAE;gBAC5C;;YAEF,MAAM,UAAU,GAAG,aAAa,CAAC,OAAO,CAAC,qBAAqB,EAAE;AAChE,YAAA,WAAW,CAAC;gBACV,GAAG,EAAE,QAAQ,CAAC,CAAC,GAAG,UAAU,CAAC,GAAG,GAAG,MAAM,CAAC,OAAO;gBACjD,IAAI,EAAE,QAAQ,CAAC,CAAC,GAAG,UAAU,CAAC,IAAI,GAAG,MAAM,CAAC,OAAO;AACpD,aAAA,CAAC;AACJ,SAAC;AAED,QAAA,MAAM,QAAQ,GAAG,QAAQ,CAAC,gBAAgB,CACxC,CAAa,UAAA,EAAA,QAAQ,CAAC,SAAS,CAAI,EAAA,CAAA,CACpC;QACD,MAAM,OAAO,GAAG,QAAQ,CAAC,QAAQ,CAAC,YAAY,CAAC;QAE/C,IAAI,OAAO,EAAE;AACX,YAAA,OAAO,CAAC,GAAG,CAAC,OAAO,CAAC;AAEpB,YAAA,aAAa,CAAC,OAAO,GAAG,OAAO;AAC/B,YAAA,YAAY,EAAE;AACd,YAAA,MAAM,CAAC,gBAAgB,CAAC,QAAQ,EAAE,YAAY,CAAC;;AAGjD,QAAA,OAAO,MAAK;YACV,IAAI,OAAO,EAAE;AACX,gBAAA,MAAM,CAAC,mBAAmB,CAAC,QAAQ,EAAE,YAAY,CAAC;;AAEtD,SAAC;AACH,KAAC,EAAE,CAAC,QAAQ,CAAC,CAAC;IAEd,MAAM,oBAAoB,GAAG,MAAK;QAChC,cAAc,CAAC,QAAQ,CAAC;AAC1B,KAAC;;IAGD,IAAI,CAAC,aAAa,IAAI,CAAC,aAAa,CAAC,OAAO,EAAE;AAC5C,QAAA,OAAO,IAAI;;AAGb,IAAA,OAAO,CAAC,GAAG,CAAC,QAAQ,EAAE,QAAQ,CAAC;;AAG/B,IAAA,OAAOC,qBAAY,CACjBC,eACE,CAAA,KAAA,EAAA,EAAA,KAAK,EAAE;AACL,YAAA,QAAQ,EAAE,UAAU;YACpB,GAAG,EAAE,QAAQ,CAAC,GAAG;YACjB,IAAI,EAAE,QAAQ,CAAC,IAAI;AACnB,YAAA,KAAK,EAAE,OAAO;AACd,YAAA,MAAM,EAAE,OAAO;AACf,YAAA,MAAM,EAAE,iBAAiB;AACzB,YAAA,eAAe,EAAE,OAAO;SACzB,EAED,QAAA,EAAA,CAAAA,eAAA,CAAA,KAAA,EAAA,EACE,KAAK,EAAE;AACL,oBAAA,OAAO,EAAE,MAAM;AACf,oBAAA,cAAc,EAAE,eAAe;AAC/B,oBAAA,UAAU,EAAE,QAAQ;AACrB,iBAAA,EAAA,QAAA,EAAA,CAEDA,eACE,CAAA,KAAA,EAAA,EAAA,QAAA,EAAA,CAAAC,cAAA,CAAA,GAAA,EAAA,EAAG,KAAK,EAAE,EAAE,QAAQ,EAAE,MAAM,EAAE,UAAU,EAAE,MAAM,EAAE,yBAAc,EAChED,eAAA,CAAA,GAAA,EAAA,EAAG,KAAK,EAAE,EAAE,QAAQ,EAAE,MAAM,EAAE,EACf,QAAA,EAAA,CAAA,cAAA,EAAA,QAAQ,CAAC,SAAS,CAAC,cAAc,EAAE,CAC9C,EAAA,CAAA,EACJC,sBAAG,IAAI,EAAE,QAAQ,CAAC,SAAS,EAAE,MAAM,EAAC,QAAQ,EAExC,QAAA,EAAA,iBAAA,EAAA,CAAA,CAAA,EAAA,CACA,EACNA,cAAA,CAAA,QAAA,EAAA,EAAQ,OAAO,EAAE,oBAAoB,EAAiB,QAAA,EAAA,QAAA,EAAA,CAAA,CAAA,EAAA,CAClD,EACNA,cAAG,CAAA,GAAA,EAAA,EAAA,KAAK,EAAE,EAAE,QAAQ,EAAE,MAAM,EAAE,YAAG,QAAQ,CAAC,OAAO,EAAA,CAAK,IAClD,EACN,QAAQ,CAAC,IAAI,CACd;AACH,CAAC;;ACnFD,SAAS,IAAI,GAAA;AACX,IAAA,MAAM,CAAC,MAAM,EAAE,SAAS,CAAC,GAAG,KAAK,CAAC,QAAQ,CAAC,KAAK,CAAC;IACjD,MAAM,EAAE,IAAI,EAAE,SAAS,EAAE,GAAG,YAAY,EAAE;AAE1C,IAAA,OAAO,CAAC,GAAG,CAAC,gBAAgB,EAAE,SAAS,CAAC;IAExC,MAAM,EAAE,MAAM,EAAE,cAAc,EAAE,GAAG,iBAAiB,EAAE;IAEtDH,eAAS,CAAC,MAAK;QACb,IAAI,MAAM,EAAE;AACV,YAAA,MAAM,CAAC,gBAAgB,CAAC,SAAS,EAAE,cAAc,CAAC;AAClD,YAAA,OAAO,MAAK;AACV,gBAAA,MAAM,CAAC,mBAAmB,CAAC,UAAU,EAAE,cAAc,CAAC;AACxD,aAAC;;AAEL,KAAC,EAAE,CAAC,MAAM,CAAC,CAAC;AAEZ,IAAA,MAAM,cAAc,GAAG,CAAC,KAAoB,KAAI;AAC9C,QAAA,IAAI,KAAK,CAAC,GAAG,KAAK,QAAQ,EAAE;YAC1B,SAAS,CAAC,KAAK,CAAC;;AAEpB,KAAC;AAED,IAAA,MAAM,WAAW,GAAG,CAAC,CAAa,KAAI;QACpC,CAAC,CAAC,eAAe,EAAE;QACnB,CAAC,CAAC,cAAc,EAAE;AAClB,QAAA,MAAM,IAAI,GAAG,CAAC,CAAC,MAA4B;AAC3C,QAAA,MAAM,SAAS,GAAG,CAAA,IAAI,aAAJ,IAAI,KAAA,SAAA,GAAA,SAAA,GAAJ,IAAI,CAAE,OAAO,CAAC,EAAE,KAAI,EAAE;QACxC,MAAM,WAAW,GAAG,QAAQ,CAAC,gBAAgB,CAC3C,CAAa,UAAA,EAAA,SAAS,CAAI,EAAA,CAAA,CAC3B;QAED,MAAM,cAAc,GAAG,KAAK,CAAC,IAAI,CAAC,WAAW,CAAC;QAE9C,MAAM,YAAY,GAAG,cAAc,CAAC,OAAO,CAAC,IAAmB,CAAC;AAEhE,QAAA,MAAM,CAAC,GAAG,CAAC,CAAC,OAAO;AACnB,QAAA,MAAM,CAAC,GAAG,CAAC,CAAC,OAAO;AACnB,QAAA,OAAO,CAAC,GAAG,CAAC,CAAC,CAAC;AACd,QAAA,MAAM,WAAW,GAAoB;YACnC,CAAC;YACD,CAAC;AACD,YAAA,OAAO,EAAE,MAAM;AACf,YAAA,IAAI,EAAE,MAAM,CAAC,QAAQ,CAAC,QAAQ;YAC9B,SAAS;YACT,YAAY;SACb;AACD,QAAA,OAAO,CAAC,GAAG,CAAC,WAAW,CAAC;QACxB,cAAc,CAAC,WAAW,CAAC;QAC3B,QAAQ,CAAC,IAAI,CAAC,SAAS,CAAC,MAAM,CAAC,gBAAgB,CAAC;QAChD,QAAQ,CAAC,mBAAmB,CAAC,OAAO,EAAE,WAAW,EAAE,IAAI,CAAC;QACxD,SAAS,CAAC,IAAI,CAAC;AACjB,KAAC;AAED,IAAA,MAAM,oBAAoB,GAAG,CAAC,CAAsC,KAAI;QACtE,CAAC,CAAC,eAAe,EAAE;QACnB,QAAQ,CAAC,IAAI,CAAC,SAAS,CAAC,GAAG,CAAC,gBAAgB,CAAC;QAC7C,QAAQ,CAAC,gBAAgB,CAAC,OAAO,EAAE,WAAW,EAAE,IAAI,CAAC;AACvD,KAAC;AAED,IAAA,QACEE,eAAA,CAAAE,mBAAA,EAAA,EAAA,QAAA,EAAA,CACED,cACE,CAAA,KAAA,EAAA,EAAA,KAAK,EAAE;AACL,oBAAA,QAAQ,EAAE,OAAO;AACjB,oBAAA,GAAG,EAAE,oBAAoB;AACzB,oBAAA,IAAI,EAAE,qBAAqB;AAC3B,oBAAA,KAAK,EAAE,OAAO;AACd,oBAAA,MAAM,EAAE,MAAM;AACd,oBAAA,MAAM,EAAE,OAAO;AACf,oBAAA,eAAe,EAAE,SAAS;AAC1B,oBAAA,KAAK,EAAE,OAAO;AACd,oBAAA,YAAY,EAAE,CAAC;AAChB,iBAAA,EACD,OAAO,EAAE,MAAM,SAAS,CAAC,CAAC,MAAM,CAAC,EAAA,QAAA,EAAA,eAAA,EAAA,CAG7B,EACNA,cAAA,CAAA,QAAA,EAAA,EACE,KAAK,EAAE;AACL,oBAAA,QAAQ,EAAE,OAAO;AACjB,oBAAA,GAAG,EAAE,qBAAqB;AAC1B,oBAAA,IAAI,EAAE,qBAAqB;AAC3B,oBAAA,KAAK,EAAE,OAAO;AACd,oBAAA,MAAM,EAAE,MAAM;AACd,oBAAA,MAAM,EAAE,OAAO;AACf,oBAAA,eAAe,EAAE,SAAS;AAC1B,oBAAA,KAAK,EAAE,OAAO;AACd,oBAAA,YAAY,EAAE,CAAC;AAChB,iBAAA,EACD,OAAO,EAAE,oBAAoB,EAAA,QAAA,EAAA,cAAA,EAAA,CAGtB,EACR,MAAM;gBACL,SAAS;gBACT;AACG,qBAAA,MAAM,CAAC,CAAC,QAAQ,KAAK,QAAQ,CAAC,IAAI,KAAK,MAAM,CAAC,QAAQ,CAAC,QAAQ;qBAC/D,GAAG,CAAC,CAAC,QAAQ,MACZA,cAAC,CAAA,YAAY,EAAmB,EAAA,QAAQ,EAAE,QAAQ,EAAA,EAA/B,QAAQ,CAAC,EAAE,CAAwB,CACvD,CAAC,CACL,EAAA,CAAA;AAEP;;ACvGA,MAAM,WAAW,GAAG,IAAIE,sBAAW,EAAE;AAErC,SAAS,GAAG,GAAA;AACV,IAAA,QACEF,cAAA,CAACG,8BAAmB,EAAA,EAAC,MAAM,EAAE,WAAW,EAAA,QAAA,EACtCH,cAAC,CAAA,IAAI,EAAG,EAAA,CAAA,EAAA,CACY;AAE1B;;;;"}
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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"}
@@ -3,5 +3,5 @@
3
3
  @tailwind utilities;
4
4
 
5
5
  .comment-cursor {
6
- cursor: url("/comment.svg"), auto !important;
6
+ cursor: url("./assets/comment.svg"), auto !important;
7
7
  }
package/package.json CHANGED
@@ -1,11 +1,15 @@
1
1
  {
2
2
  "name": "galadrim-feedback",
3
3
  "private": false,
4
- "version": "0.0.2",
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,3 +0,0 @@
1
- export const Avatar = ({ avatar }: { avatar: string }) => {
2
- return <img src={avatar} className="w-10 h-10 rounded-full" />;
3
- };
@@ -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,7 +0,0 @@
1
- @tailwind base;
2
- @tailwind components;
3
- @tailwind utilities;
4
-
5
- .comment-cursor {
6
- cursor: url("/comment.svg"), auto !important;
7
- }
@@ -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
- };
@@ -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
- };
@@ -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;
@@ -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;
@@ -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
- };
@@ -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
- }
@@ -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
- }