{"version":3,"file":"List-Db-FAhpI.js","sources":["../../../node_modules/@mui/icons-material/esm/Chat.js","../../../node_modules/@mui/icons-material/esm/EventOutlined.js","../../../node_modules/@mui/icons-material/esm/LabelOutlined.js","../../../node_modules/@mui/icons-material/esm/PlaceOutlined.js","../../../frontend/src/Components/Posts/Shared/PostTags/PostTags.tsx","../../../frontend/src/Components/Posts/PostTagsBuilder/PostTagsBuilder.tsx","../../../frontend/src/Components/DesignSystem/PostDescription/PostDescription.tsx","../../../frontend/src/Components/DesignSystem/TranslationButton/TranslationButton.tsx","../../../frontend/src/Hooks/useTranslation.tsx","../../../frontend/src/Components/InlineCommentForm/InlineCommentForm.tsx","../../../frontend/src/Components/Comments/Show/Comment/Comment.tsx","../../../frontend/src/Components/Comments/Show/StatusChange/StatusChange.tsx","../../../frontend/src/Components/Comments/Show/List/List.tsx"],"sourcesContent":["\"use client\";\n\nimport createSvgIcon from \"./utils/createSvgIcon.js\";\nimport { jsx as _jsx } from \"react/jsx-runtime\";\nexport default createSvgIcon(/*#__PURE__*/_jsx(\"path\", {\n d: \"M20 2H4c-1.1 0-1.99.9-1.99 2L2 22l4-4h14c1.1 0 2-.9 2-2V4c0-1.1-.9-2-2-2M6 9h12v2H6zm8 5H6v-2h8zm4-6H6V6h12z\"\n}), 'Chat');","\"use client\";\n\nimport createSvgIcon from \"./utils/createSvgIcon.js\";\nimport { jsx as _jsx } from \"react/jsx-runtime\";\nexport default createSvgIcon(/*#__PURE__*/_jsx(\"path\", {\n d: \"M19 4h-1V2h-2v2H8V2H6v2H5c-1.11 0-1.99.9-1.99 2L3 20c0 1.1.89 2 2 2h14c1.1 0 2-.9 2-2V6c0-1.1-.9-2-2-2m0 16H5V10h14zm0-12H5V6h14zm-7 5h5v5h-5z\"\n}), 'EventOutlined');","\"use client\";\n\nimport createSvgIcon from \"./utils/createSvgIcon.js\";\nimport { jsx as _jsx } from \"react/jsx-runtime\";\nexport default createSvgIcon(/*#__PURE__*/_jsx(\"path\", {\n d: \"M17.63 5.84C17.27 5.33 16.67 5 16 5L5 5.01C3.9 5.01 3 5.9 3 7v10c0 1.1.9 1.99 2 1.99L16 19c.67 0 1.27-.33 1.63-.84L22 12zM16 17H5V7h11l3.55 5z\"\n}), 'LabelOutlined');","\"use client\";\n\nimport createSvgIcon from \"./utils/createSvgIcon.js\";\nimport { jsx as _jsx } from \"react/jsx-runtime\";\nexport default createSvgIcon(/*#__PURE__*/_jsx(\"path\", {\n d: \"M12 12c-1.1 0-2-.9-2-2s.9-2 2-2 2 .9 2 2-.9 2-2 2m6-1.8C18 6.57 15.35 4 12 4s-6 2.57-6 6.2c0 2.34 1.95 5.44 6 9.14 4.05-3.7 6-6.8 6-9.14M12 2c4.2 0 8 3.22 8 8.2 0 3.32-2.67 7.25-8 11.8-5.33-4.55-8-8.48-8-11.8C4 5.22 7.8 2 12 2\"\n}), 'PlaceOutlined');","import React from 'react';\nimport PropTypes from 'prop-types';\nimport { has } from 'lodash';\nimport { Box, Tooltip, Typography, Link } from '@mui/material';\nimport { PlaceOutlined as MapsIcon, EventOutlined as EventIcon } from '@mui/icons-material';\nimport { FormattedMessage, useIntl } from 'react-intl';\n\nimport { moment } from '../../../../I18n';\nimport useWidth from '../../../../Hooks/useWidth';\nimport LocationLink from '../../../Maps/LocationLink';\n\nfunction Tag({ icon, label }: any) {\n return (\n \n {icon}\n {label}\n \n );\n}\n\nfunction PostTags({ items }: any) {\n const intl = useIntl();\n\n const renderItem = (id: any, item: any) => {\n const hasLocation = has(item, 'lat') && has(item, 'lng');\n\n if (hasLocation) {\n return } label={} />;\n }\n\n if (['start_at', 'end_at'].every((key) => Object.keys(item).includes(key))) {\n let periodI18nStringID;\n\n if (moment(item.start_at).isSame(item.end_at, 'day')) {\n periodI18nStringID = 'POSTS.TAGS.PERIOD_SAME_DAY';\n } else {\n periodI18nStringID = useWidth('down', 'sm')\n ? 'POSTS.TAGS.PERIOD_SHORT'\n : 'POSTS.TAGS.PERIOD';\n }\n\n return (\n }\n label={\n \n {chunks},\n }}\n />\n \n }\n />\n );\n }\n\n const hasTooltip = has(item, 'tooltip');\n if (hasTooltip) {\n return (\n \n {item.label}\n \n }\n />\n );\n }\n\n return (\n \n \n {item.label}\n \n \n ) : (\n {item.label}\n )\n }\n />\n );\n };\n\n return (\n \n {items.map((item: any, index: any) => renderItem(index, item))}\n \n );\n}\n\nPostTags.propTypes = {\n items: PropTypes.arrayOf(\n PropTypes.oneOfType([\n // Generic item\n PropTypes.shape({\n label: PropTypes.string.isRequired,\n icon: PropTypes.element.isRequired,\n url: PropTypes.string,\n }),\n\n // Location item\n PropTypes.shape({\n label: PropTypes.string.isRequired,\n lat: PropTypes.string,\n lng: PropTypes.string,\n }),\n\n // Date item\n PropTypes.shape({\n start_at: PropTypes.string.isRequired,\n end_at: PropTypes.string.isRequired,\n }),\n ]),\n ).isRequired,\n};\n\nexport default PostTags;\n","import React from 'react';\nimport { useIntl } from 'react-intl';\nimport { LabelOutlined as CategoryIcon } from '@mui/icons-material';\nimport PropTypes from 'prop-types';\n\nimport PostTags from '../Shared/PostTags';\nimport CustomIcon from '../../DesignSystem/CustomIcon';\nimport { Icons } from '../../../Themes';\nimport { formatLabel } from '../../../Utils/Post/budget';\nimport { Groups } from '../../../Utils/Types';\n\nfunction PostTagsBuilder({ post, space }: any) {\n const intl = useIntl();\n const items = [];\n const [lat, lng] = (post.latlng || '').split(',');\n\n if (post.budget) {\n items.push({\n label: formatLabel(intl, post.budget, space.currency.key),\n icon: ,\n });\n }\n\n const tagNames = post?.included?.tags?.map((tag: any) => tag.label);\n const tagsCount = tagNames?.length || 0;\n if (tagsCount > 0) {\n if (tagsCount > 3) {\n items.push({\n label: `${tagNames.slice(0, 3).join(', ')} ${intl.formatMessage(\n { id: 'POSTS.DETAILS.MORE_TAGS' },\n { count: tagsCount - 3 },\n )}`,\n icon: ,\n tooltip: tagNames.slice(3).join(', '),\n });\n } else {\n items.push({\n label: tagNames.join(', '),\n icon: ,\n });\n }\n }\n\n if (!Groups.ADMINS.includes(post?.included?.owner?.role) && post?.authored_by_first_name) {\n items.push({\n label: intl.formatMessage(\n { id: 'POSTS.DETAILS.AUTHORED_BY' },\n { name: post.authored_by_first_name },\n ),\n icon: ,\n });\n }\n\n if (post.location) {\n items.push({ label: post.location, lat, lng });\n }\n\n if (post.event_url) {\n items.push({\n label: intl.formatMessage({ id: 'POSTS.DETAILS.EVENT.EVENT_URL' }),\n url: post.event_url,\n icon: ,\n });\n }\n\n return (\n <>\n {post.start_at && }\n\n \n \n );\n}\n\nPostTagsBuilder.propTypes = {\n space: PropTypes.object.isRequired,\n post: PropTypes.shape({\n end_at: PropTypes.string,\n event_url: PropTypes.string,\n latlng: PropTypes.string,\n location: PropTypes.string,\n start_at: PropTypes.string,\n }),\n};\n\nexport default PostTagsBuilder;\n","import React from 'react';\nimport PropTypes from 'prop-types';\nimport { Typography, Box } from '@mui/material';\n\nimport ReadMore from '../../ReadMore/ReadMore';\n\nfunction PostDescription({ content, readMore = false, subtitle = null }: any) {\n return (\n
\n \n {subtitle && (\n \n {subtitle}\n \n )}\n\n \n \n
\n );\n}\n\nPostDescription.propTypes = {\n parent: PropTypes.object,\n subtitle: PropTypes.string,\n content: PropTypes.string.isRequired,\n readMore: PropTypes.bool,\n};\n\nexport default PostDescription;\n","import React from 'react';\nimport { Link as ButtonLink } from '@mui/material';\nimport { FormattedMessage } from 'react-intl';\n\nfunction TranslationButton({ loading, translated, onClick }: any) {\n const buttonText = translated ? 'TRANSLATION.REVERSE_TRANSLATE' : 'TRANSLATION.TRANSLATE';\n\n return (\n \n \n \n \n \n );\n}\n\nexport default TranslationButton;\n","import React from 'react';\nimport { useIntl } from 'react-intl';\n\nimport TranslationButton from '../Components/DesignSystem/TranslationButton';\nimport useApiQuery from './useApiQuery';\nimport queriesConfig from '../Services/queriesConfig';\nimport Analytics from '../Services/AnalyticsService';\nimport Note from '../Utils/Types/Note';\n\nexport function useCommentTranslation(comment: Note, postId: any) {\n const intl = useIntl();\n const targetLang = intl.locale;\n const [fetchTranslation, setFetchTranslation] = React.useState(false);\n const [isShown, setIsShown] = React.useState(false);\n\n Analytics.track('Comment - Translation', {\n 'Post ID': postId,\n 'Comment ID': comment.id,\n fromLanguage: comment.lang,\n toLanguage: targetLang,\n });\n\n const { data, isLoading } = useApiQuery({\n onSuccess() {\n setIsShown(true);\n },\n query: queriesConfig.getCommentTranslation(comment?.id, targetLang),\n options: {\n enabled: fetchTranslation,\n },\n });\n\n const translatedContent = data ? (data as { content: string }).content : undefined;\n\n const translateButton = comment.lang !== targetLang && (\n {\n setFetchTranslation(!translatedContent);\n if (translatedContent) {\n setIsShown(!isShown);\n }\n }}\n loading={isLoading}\n translated={Boolean(translatedContent) && isShown}\n />\n );\n\n const contentToShow = isShown ? translatedContent : comment.content;\n\n return {\n translateButton,\n content: contentToShow,\n };\n}\n","import React from 'react';\nimport { Box, Grid2 as Grid } from '@mui/material';\nimport { forOwn, get, has } from 'lodash';\nimport { Controller, useForm } from 'react-hook-form';\nimport { FormattedMessage, useIntl } from 'react-intl';\nimport { useNavigate } from 'react-router-dom';\n\nimport { CloudinaryConfig, cloudinaryImageUrl } from '../../Services/Cloudinary';\nimport ButtonWithLoader from '../ButtonWithLoader';\nimport Avatar from '../DesignSystem/Avatar';\nimport { TextFieldDefault } from '../Posts/Forms';\nimport useSpace from '../../Hooks/useSpace';\nimport { loginPath } from '../../Utils/routes';\nimport { validateCanComment } from '../../Utils/Post/securedStrategies';\nimport { isIdea } from '../../Utils/Post';\nimport useApiMutation from '../../Hooks/useApiMutation';\nimport queriesConfig from '../../Services/queriesConfig';\nimport { useModal } from 'mui-modal-provider';\nimport EntitySubscriptionModal from '../EntitySubscriptionModal';\nimport useCurrentUser from '../../Hooks/useCurrentUser';\n\nexport default function InlineCommentForm({\n post,\n commentId,\n onSubmit,\n autoFocus = false,\n parentPost,\n discussionType = 'public', // Type can be 'public' or 'private' or 'email'\n}: any) {\n const postId = post.id;\n const { space } = useSpace();\n const navigate = useNavigate();\n const { currentUser, isLoggedIn } = useCurrentUser();\n const { showModal } = useModal();\n const intl = useIntl();\n const commentPlaceHolder = intl.formatMessage({\n id:\n discussionType === 'private'\n ? 'DISCUSSION.SHOW.COMMENT_PLACEHOLDER.PRIVATE'\n : 'DISCUSSION.SHOW.COMMENT_PLACEHOLDER.PUBLIC',\n });\n\n const {\n control,\n handleSubmit,\n setError,\n reset,\n formState: { errors },\n } = useForm({\n mode: 'onSubmit',\n criteriaMode: 'all',\n defaultValues: {\n content: '',\n },\n });\n\n const afterSubmit = () => {\n reset();\n onSubmit();\n };\n\n let endpoint;\n if (discussionType === 'private' || discussionType === 'email') {\n endpoint = queriesConfig.postDiscussionNote(postId, {\n parentId: commentId,\n type: discussionType,\n });\n } else if (commentId) {\n endpoint = queriesConfig.postCommentForComment(postId, commentId);\n } else {\n endpoint = queriesConfig.postComment(postId);\n }\n\n const { mutate: createNote, isLoading } = useApiMutation({\n query: endpoint,\n onSuccess: afterSubmit,\n onError: (apiErrors: any) => {\n forOwn(apiErrors, (message) => {\n setError('content', message);\n });\n },\n });\n\n const handleClick = (event: Event) => {\n event.preventDefault();\n\n if (!isLoggedIn) {\n navigate(loginPath());\n } else {\n if (!space.entity_subscription) {\n showModal(EntitySubscriptionModal);\n } else {\n validateCanComment(isIdea(post.type) ? parentPost : post);\n }\n }\n };\n\n return (\n
\n \n \n \n \n \n \n \n \n (\n \n )}\n />\n \n \n \n \n \n \n \n \n
\n );\n}\n","import React, { useState } from 'react';\nimport { FormattedMessage } from 'react-intl';\nimport { styled } from '@mui/material/styles';\nimport ReplyIcon from '@mui/icons-material/Reply';\nimport { Grid2 as Grid, Paper, Typography, Box, Link as LinkButton, Button } from '@mui/material';\nimport { get } from 'lodash';\n\nimport CardHeader from '../../../DesignSystem/CardHeader';\nimport MenuActions from '../../../DesignSystem/MenuActions';\nimport { useCommentTranslation } from '../../../../Hooks/useTranslation';\nimport useWidth from '../../../../Hooks/useWidth';\nimport { cloudinaryImageUrl, CloudinaryConfig } from '../../../../Services/Cloudinary';\nimport InlineCommentForm from '../../../InlineCommentForm';\nimport AttachmentLinksList from '../../../Posts/Shared/AttachmentLinksList';\nimport CommentsList from '../List/List';\nimport useApiQuery from '../../../../Hooks/useApiQuery';\nimport queriesConfig from '../../../../Services/queriesConfig';\n\nconst StyledContainer = styled(Paper, {\n shouldForwardProp: (prop: any) =>\n !['isNested', 'isDeleted', 'isMobile', 'isCreatedByAdminOrFluimaster'].includes(prop),\n})(({ theme, isNested, isDeleted, isMobile, isCreatedByAdminOrFluimaster }: any) => {\n if (isNested) {\n return {\n marginTop: theme.spacing(3),\n marginBottom: 0,\n marginLeft: isMobile ? 0 : theme.spacing(4),\n paddingBottom: theme.spacing(3),\n border: 'none',\n };\n }\n\n if (isDeleted) {\n return {\n padding: theme.spacing(2),\n paddingLeft: theme.spacing(3),\n };\n }\n\n return {\n padding: theme.spacing(3),\n ...(isCreatedByAdminOrFluimaster && {\n boxShadow: `inset 4px 0px 0px ${theme.palette.secondary.main}, ${theme.shadows[1]}`,\n }),\n };\n});\n\nfunction Comment({\n comment,\n authorId,\n post,\n parentPost,\n isNested,\n disabled = false,\n censored = true,\n html = false,\n discussionType,\n onDestroy = () => {},\n}: any) {\n const owner = get(comment, 'included.owner', {});\n const author = get(comment, 'included.author', {});\n const isMobile = useWidth('down', 'sm');\n const isCreatedByAdminOrFluimaster = comment.is_admin;\n const isAuthor = authorId === owner.id && post.authored_by !== 'a_third_party';\n const isDeleted = !!comment.deleted_at;\n const canDeleteComment = comment.permissions.can_delete_comment;\n const canReportComment = comment.permissions.can_report_comment;\n\n const discussionTypeFromComment = comment.discussion_type || discussionType;\n\n const { translateButton, content } = useCommentTranslation(comment, post.id);\n const [displayConversationThread, setDisplayConversationThread] = useState(false);\n\n const queryParams = {\n page: 'all',\n limit: 100,\n note_id: comment.id,\n parent_id: comment.id,\n type: discussionTypeFromComment,\n };\n\n const { data: subComments, refetch } = useApiQuery({\n query: queryParams.type\n ? queriesConfig.getDiscussionComments(post.id, queryParams)\n : queriesConfig.getComments(post.id, queryParams),\n options: {\n enabled: displayConversationThread,\n },\n });\n\n return (\n \n {isDeleted ? (\n \n \n \n ) : (\n <>\n \n \n }\n avatarSize={40}\n key={comment.id}\n />\n \n\n \n \n \n\n \n \n \n\n \n\n {!displayConversationThread && comment.comments_count > 0 && (\n \n setDisplayConversationThread(true)}\n startIcon={}\n size=\"small\"\n >\n \n \n \n )}\n\n {displayConversationThread && (\n <>\n \n \n \n\n