\n \n \n \n \n {/* PM Name's or Team Name's Product Shape Overview */}\n {firstName !== AGGREGATED_SUBMISSION &&\n lastName !== AGGREGATED_SUBMISSION\n ? `${firstName} ${lastName}`\n : `${teamName}`}\n {`'s Product Shape Overview`}\n \n \n \n \n Thank you for taking the time to complete the survey! Your results\n have been used to create the product shape below.\n The Product Shape is a tool for product managers to\n evaluate their own talents and expertise across various areas.\n \n \n {!saving && !isMobile && (\n setSaving(true)}\n style={{\n fontWeight: 'bold',\n width: '300px',\n marginBottom: '4rem',\n borderRadius: 16,\n }}\n disableRipple\n >\n Download Survey results\n \n )}\n {/* Product shape, areas of growth and tooltip: */}\n \n \n \n Product Shape\n {!saving && }\n \n \n\n \n \n \n \n \n \n \n \n \n = 900 ? '4rem' : ''}>\n \n {window.innerWidth >= 600\n ? `Current Strong Area`\n : `Current\\nStrong Area`}\n \n \n {findMinMax(jsonData.categories, scores, false).name}\n \n \n \n \n = 900 ? '4rem' : ''}>\n \n {window.innerWidth >= 600\n ? `Current Growth Area`\n : `Current\\nGrowth Area`}\n \n \n {findMinMax(jsonData.categories, scores, true).name}\n \n \n \n \n \n \n \n\n {/* Shape breakdown table: */}\n \n \n Product Shape Breakdown\n \n\n = 900 ? 'left' : 'center'}\n marginBottom='20px'\n flexWrap='wrap'\n >\n \n Legend\n \n \n = 600 ? '14px' : '10px',\n }}\n />\n = 600 ? '14px' : '10px',\n }}\n />\n = 600 ? '14px' : '10px',\n }}\n />\n \n \n {/* Add space btween table and bottom of page */}\n \n {/* Calls results table function to create table removed userAnswers and questions because only really need scores */}\n {generateAnswersTable(\n [\n 'Product Design &\\nExecution',\n 'User &\\nExperience',\n 'Business &\\nMarketing',\n 'Leadership &\\nManagement',\n ],\n scores,\n window.innerWidth <= 900\n )}\n \n \n \n
\n \n );\n};\n\nexport default ClientProductShape;\n","import { useState, useEffect, CSSProperties } from 'react';\nimport {\n Chart as ChartJS,\n RadialLinearScale,\n PointElement,\n LineElement,\n Filler,\n Tooltip,\n Legend,\n} from 'chart.js';\nimport { Radar } from 'react-chartjs-2';\nimport {\n Box,\n Grid,\n Typography,\n Link,\n Stack,\n Modal,\n CircularProgress,\n} from '@mui/material';\nimport { saveAs } from 'file-saver';\nimport { StyledButton } from '@/components/styled/button';\nimport { FormPublicFieldFragment } from 'graphql/fragment/form.public.fragment';\nimport MenuBookTwoToneIcon from '@mui/icons-material/MenuBookTwoTone';\nimport OpenInNewIcon from '@mui/icons-material/OpenInNew';\nimport { lighten } from 'polished';\nimport html2canvas from 'html2canvas';\nimport {\n generateAnswersTable,\n generateTableLegend,\n} from '../pillars-of-product/components/ResultsTable';\nimport Supporter from '../pillars-of-product/components/Supporter';\nimport DerikJones from './assets/derik-jones.jpg';\nimport AshleyWillis from './assets/ashley-willis.jpg';\n\ninterface UserAnswer {\n field: string; // question id\n data: string; // user chosen question answer\n}\n\nconst DesignCapabilitiesShape = ({\n scores,\n design,\n jsonData,\n questions,\n userAnswers,\n}: any) => {\n ChartJS.register(\n RadialLinearScale,\n PointElement,\n LineElement,\n Filler,\n Tooltip,\n Legend,\n {\n id: 'custom_canvas_background_color',\n beforeDraw: (chart) => {\n const ctx = chart.canvas.getContext('2d');\n ctx!.save();\n ctx!.globalCompositeOperation = 'destination-over';\n ctx!.fillStyle = '#000000';\n ctx!.fillRect(0, 0, chart.width, chart.height);\n ctx!.restore();\n },\n }\n );\n\n const [saving, setSaving] = useState(false);\n\n const saveResults = async () => {\n const endPage = document.getElementById('endpage-div');\n const canvas = await html2canvas(endPage!);\n canvas.toBlob(function (blob) {\n saveAs(blob!, 'design-capabilities-chart.png');\n });\n };\n\n // used to show saving modal for a second, without it modal just flashes in and out\n const stopSaving = async () => {\n const sleep = (ms: number) => new Promise((r) => setTimeout(r, ms));\n await sleep(1000);\n setSaving(false);\n };\n\n useEffect(() => {\n if (saving) {\n saveResults();\n stopSaving();\n }\n }, [saving]);\n\n const modalStyling: CSSProperties = {\n position: 'absolute' as 'absolute',\n top: '50%',\n left: '50%',\n transform: 'translate(-50%, -50%)',\n border: `2px solid ${design.colors.question}`,\n borderRadius: '8px',\n backgroundColor: design.colors.background,\n color: design.colors.button,\n padding: '40px',\n };\n\n const saveChart = () => {\n //save to png\n const canvasSave = document.getElementById('pillars-of-product-chart');\n // @ts-ignore: Object is possibly 'null'.\n canvasSave.toBlob(function (blob) {\n saveAs(blob, 'pillars-of-product-chart.png');\n });\n };\n\n // console.log(chart.config.options) to see this object\n const chartData = {\n labels: jsonData.chartCategories,\n datasets: [\n {\n label: 'Shape',\n data: scores,\n backgroundColor: 'rgba(134, 188, 37, 0.2)',\n borderColor: 'rgba(134, 188, 37, 1)',\n color: 'rgba(255, 255, 255, 0.75)',\n borderWidth: 1,\n },\n ],\n };\n\n // console.log(chart.config.options) to see this object\n const chartOptions: {\n plugins: {\n legend: {\n display: boolean;\n };\n };\n scales: {\n r: {\n min: number;\n max: number;\n ticks: {\n stepSize: number;\n callback: any;\n };\n grid: {\n color: string;\n lineWidth: number;\n };\n pointLabels: any;\n };\n };\n } = {\n plugins: {\n legend: {\n display: false,\n },\n },\n scales: {\n r: {\n min: 0,\n max: 3,\n ticks: {\n stepSize: 1,\n callback: function (index: number): string {\n switch (index) {\n case 1:\n return 'Lagging';\n case 2:\n return 'Sustainable';\n case 3:\n return 'Leading';\n default:\n return '';\n }\n },\n },\n grid: {\n color: 'rgba(255, 255, 255, 0.75)',\n lineWidth: 2,\n },\n pointLabels: {\n /* Adjust data label font size according to chart size */\n font: function (context: { chart: { width: any } }) {\n var width = context.chart.width;\n var size = Math.round(width / 48);\n return {\n size: size,\n };\n },\n color: 'rgba(255, 255, 255, 1)',\n },\n },\n },\n };\n\n if (userAnswers.length !== 21) {\n return (\n
\n Something is wrong with the submission. The amount of answers don't\n match the amount of questions.\n
\n );\n }\n\n return (\n
\n \n \n \n \n \n
\n \n \n \n Thank you for completing your Design Capabilities Assessment,{' '}\n {userAnswers[0].data.replace(/\\\"/g, '')}.\n \n \n \n \n Thank you for taking the time to complete our survey on the 4\n pillars of product-centric organizational design. Your inputs have\n been scored across Deloitte's 16 dimensions of product centricity\n and ranked as Lagging, Leading, or Sustaining.\n {'\\n\\n'}This scoring is focused on delivering exceptional products\n and services to your customers. We appreciate your willingness to\n share your insights and experiences with us, and to learn more\n about Deloitte's 4 Pillars of Product Management please read our\n white paper{' '}\n {saving ? null : (\n \n here\n \n \n )}\n {'.'}\n \n \n \n \n Design Capabilities Assessment Details\n \n \n {generateAnswersTable(\n questions,\n [\n 'Process &\\nTools',\n 'Talent &\\nOrganization',\n 'Craft &\\nStrategy',\n 'Impact &\\nFuture',\n ],\n userAnswers,\n window.innerWidth <= 900\n )}\n \n \n Scoring is defined as follows:\n \n \n \n {generateTableLegend([\n 'Current operations fall below outlined standards compared to market competitors and require material structural adjustments.',\n 'Current operations are in line with outlined standards compared to market competitors but have room for growth and improvement.',\n 'Current operations exceed outlined standards compared to market competitors and differentiate from competition.',\n ])}\n \n \n \n Your next steps\n \n \n \n \n \n \n \n Now that you have shared your insights and experiences with\n us, we invite you to take the next step and join us in our\n efforts to create more product-centric organizations. Please\n reach out to learn more about how Deloitte is leading\n organizational changes to become more product-centric.\n \n \n \n \n {saving ? null : (\n \n \n Contact\n \n setSaving(true)}\n style={{\n fontWeight: 'bold',\n width: '75%',\n }}\n disableRipple\n >\n SAVE YOUR RESULTS\n \n \n )}\n \n \n \n \n \n \n \n \n \n \n \n \n For more information, please contact us:\n \n \n \n \n \n \n \n \n \n
\n \n {loading && (\n \n \n \n )}\n \n \n \n Download Report\n \n \n \n \n \n \n\n \n \n Readiness Score\n \n \n {calculatedScores\n ? Math.floor(\n calculatedScores.aggregatedScore.overall_aggregated_score\n )\n : 0}\n \n \n \n\n \n Thank you for completing the GenAI Readiness Assessment, {firstName}\n .\n \n {calculatedScores && scoringLabels && (\n <>\n \n Your genAI ambitions are focused on{' '}\n {selectedUseCase.toLowerCase()} by{' '}\n {selectedDevelopmentPattern.toLowerCase()}. Based on your\n responses, we've calculated your score to be{' '}\n {Math.floor(\n calculatedScores.aggregatedScore.overall_aggregated_score\n )}{' '}\n (out of 100). {overallStatement}. However, there are several\n opportunities to upscale your maturity. We can help you maximize\n the impact of further investment to mature your genAI capability\n by focusing on the the following areas:{' '}\n {improvementAreas ? improvementAreas : ''}\n \n \n \n \n {scoringLabels?.technology_and_tooling}\n \n\n \n \n Technology & Tooling\n \n \n\n \n \n Not all platforms are “AI-Ready” out of the box, and not\n all technologies can support the heavy data needs and\n processing complexities of a genAI project at scale.\n Understanding how ready your existing technology\n infrastructure is for genAI is critical to mitigating\n compatibility challenges and potential security concerns.\n To be genAI-ready, consider the following:\n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n {scoringLabels.technology_and_tooling === 'Strong' && (\n \n You've done the work to ensure your technology can\n successfully support a genAI initiative. Plan now for\n the next wave of technology transformation by staying on\n top of emerging technology and cyber risks/requirements.\n \n )}\n {scoringLabels.technology_and_tooling === 'Moderate' && (\n \n You have some of the right infrastructure in place to\n support your genAI ambitions, but the gaps could leave\n you vulnerable as you grow. Clean up your technical debt\n and be sure you have invested in the right set of\n platforms to implement, operate, and scale your genAI\n ambitions.\n \n )}\n {scoringLabels.technology_and_tooling === 'Lacking' && (\n \n Before you launch a genAI project you will need to make\n some key updates to your enterprise technology to ensure\n you can execute your ambitions in a secure and scalable\n way. Modernize your marketing technology stack to help\n ensure your genAI project has the right support to\n succeed.\n \n )}\n \n \n \n \n {scoringLabels?.enterprise_readiness}\n \n\n \n \n Enterprise Readiness\n \n \n \n \n The true potential of genAI is harnessed when it is\n successfully integrated with other technologies and\n business processes and thoroughly embraced and adopted by\n employees, leaders, and customers alike. To get\n genAI-ready, consider the following:\n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n {scoringLabels.enterprise_readiness === 'Strong' && (\n \n Your organization has implemented a strong foundation of\n processes and talent to implement and operate your genAI\n strategy. Implementing the right feedback mechanisms for\n workforce and customers will help ensure you catch\n emerging weak spots before they endanger your project.\n \n )}\n {scoringLabels.enterprise_readiness === 'Moderate' && (\n \n You have some of the right capabilities to help manage\n the change that genAI will bring. A change management\n plan and roadmap can help you identify weak spots where\n new capacities may be needed and help ensure you have\n budgeted the right time and resources to execute your\n ambitions fully.\n \n )}\n {scoringLabels.enterprise_readiness === 'Lacking' && (\n \n Your project needs a firm foundation to succeed. Before\n investing heavily in a genAI pilot, be sure you clearly\n understand how your proposed project will affect all\n parts of your organization from front line to back\n office. Consider filling in the gaps in your core\n capabilities through in-housing, outsourcing, or\n strategic partnerships that can help you rapidly scale\n while you build your teams' capacity and expertise.\n \n )}\n \n \n \n \n {scoringLabels?.data_maturity}\n \n \n \n Data & Analytics\n \n \n \n \n Successful genAI projects rely on large amounts of data to\n work their magic. To be useful, that data must be\n trustworthy, secure, accessible, and organized so that\n your genAI tool can successfully interpret for meaningful\n insights and outputs. To get genAI-ready, consider the\n following:\n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n {scoringLabels.data_maturity === 'Strong' && (\n \n Your data infrastructure is integrated, well organized,\n and ready for implementing a genAI project. Plan now for\n the potential growth in data storage and support needs\n as your project scales.\n \n )}\n {scoringLabels.data_maturity === 'Moderate' && (\n \n You have some of the right data infrastructure in place,\n but your project may get stalled as your need for\n accurate data increases. Fix the gaps in your customer\n data infrastructure to help ensure your genAI outputs.\n \n )}\n {scoringLabels.data_maturity === 'Lacking' && (\n \n Before you launch your genAI ambitions, you will want to\n clearly map what data you will need and where it is\n currently stored within your organization. A single\n customer data platform can provide the right foundation\n to collect, clean, organize, and shape your data into a\n format that can be dynamically used by your genAI\n project.\n \n )}\n \n \n \n \n {scoringLabels?.strategy_and_governance}\n \n \n \n Strategy & Governance\n \n \n \n \n Trustworthiness and effectiveness are two of the most\n critical factors in determining whether your genAI project\n will create an impact that matters to your bottom line. To\n be genAI-ready you should consider the following:\n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n\n {scoringLabels.strategy_and_governance === 'Strong' && (\n \n You have clear governance structures in place to ensure\n this new technology is deployed safely and in alignment\n with your company's legal and risk profile and brand\n reputation. You've identified a use case that solves\n a key pain point for your organization in a way that\n will build trust with both your customers and your\n workforce. Maintaining a Human in the Loop governance\n for new features as they are developed will help ensure\n you can spot and mitigate new risks (and capitalize on\n new opportunities) as they emerge.\n \n )}\n {scoringLabels.strategy_and_governance === 'Moderate' && (\n \n You have some of the right data infrastructure in place,\n but your project may get stalled as your need for\n accurate data increases. Fix the gaps in your customer\n data infrastructure to ensure your genAI outputs'\n stability, safety, and accuracy.\n \n )}\n {scoringLabels.strategy_and_governance === 'Lacking' && (\n \n Before you launch your genAI ambitions, you will want to\n clearly map the data you need and where it is currently\n stored within your organization. A single customer data\n platform can provide the right foundation to collect,\n clean, organize, and shape your data into a format that\n your genAI project can dynamically use.\n \n )}\n \n \n \n >\n )}\n {graphData.length > 0 && (\n <>\n \n How do you compare to your peers?\n \n \n See how your answers compare to the sample group of participants\n that have been surveyed for this questionnaire. The highlighted\n numbers show your selected answers; how do you measure up?\n \n \n {graphData.map((item) => (\n \n ))}\n \n \n \n Get Started On Your GenAI Roadmap - ddbiz@deloitte.com\n \n \n >\n )}\n \n
\n >\n );\n};\n\nexport default GenAiReadiness;\n","import { FormPublicFieldFragment } from '../../../../graphql/fragment/form.public.fragment';\nimport { Typography, Grid, Stack, Box } from '@mui/material';\nimport SentimentVeryDissatisfiedIcon from '@mui/icons-material/SentimentVeryDissatisfied';\nimport SentimentNeutralIcon from '@mui/icons-material/SentimentNeutral';\nimport SentimentVerySatisfiedIcon from '@mui/icons-material/SentimentVerySatisfied';\n\ninterface UserAnswer {\n field: string; // question id\n data: string; // user chosen question answer\n}\n\nexport const generateAnswersTable = (\n questions: FormPublicFieldFragment[],\n pillars: Array,\n userAnswers: Array,\n mobile: boolean\n) => {\n const focusAreaLevels = ['Lagging', 'Sustainable', 'Leading'];\n const pillarGridItemClass = {\n backgroundColor: '#86BC25',\n padding: '20px',\n };\n const pillarClass = {\n whiteSpace: 'pre-line',\n fontSize: '1.5rem',\n fontWeight: 'bold',\n };\n /* \n on desktop, a \"row\" on the grid is rendered as it sounds like\n row 0 - v&s, t&o, f&o, d&m\n row 1 - focusAreaClasses[0], focusAreaClasses[1], focusAreaClasses[2], focusAreaClasses[3], \n row 2 - 0, 1, 2, 3\n row 3 - 0, 1, 2, 3\n row 4 - 0, 1, 2, 3\n\n on mobile, a \"row\" on the grid is rendered as a pillar's column\n row 0 - v&s\n focusAreaClasses[0]\n focusAreaClasses[1]\n focusAreaClasses[2]\n focusAreaClasses[3]\n \n row 1 - t&o\n 0\n 1\n 2\n 3\n etc\n */\n const focusAreaClasses = [\n { backgroundColor: 'rgba(255, 255, 255, 0.95)' },\n { backgroundColor: 'white' },\n { backgroundColor: 'rgba(255, 255, 255, 0.95)' },\n { backgroundColor: 'white' },\n ];\n // lagging, sustainable, leading chip colors\n const focusAreaChipClasses = [\n {\n backgroundColor: '#FFCCCC',\n color: 'rgba(133, 0, 0, 0.87)',\n border: '1px solid rgba(133, 0, 0, 0.87)',\n },\n {\n backgroundColor: '#FAF3B9',\n color: 'rgba(255, 152, 0, 1)',\n border: '1px solid rgba(255, 152, 0, 1)',\n },\n {\n backgroundColor: '#E8F9F1',\n color: 'rgba(0, 111, 59, 1)',\n border: '1px solid rgba(0, 111, 59, 1)',\n },\n ];\n const tableItems: Array = [<>>, <>>, <>>, <>>, <>>];\n /*\n firstCellIndex - starting point of the column or row in the answers table\n starting from question 5 to +3, those questions belong into a specific pillar and the table should look like this:\n\n vision & strategy - 5 6 7 8\n talent & organization - 9 10 11 12\n focus on users & outcomes - 13 14 15 16\n delivery & mindset - 17 18 19 20\n\n on mobile, we render each pillar with its questions as its own row since mobile do not have space for 4 \"wide\" rows\n thus, we want each pillar (ex vision & strategy) to have all it's focus areas (ex 5 6 7 8) in a row\n and the table should look like this:\n\n V&S scroll down -> T&O scroll down -> F&O scroll down -> D&M\n 5 9 13 17\n 6 10 14 18\n 7 11 15 19\n 8 12 16 20\n \n in this case, the answers table has 4 rows that are really \"tall but short width wise\"\n\n on desktop, we have space for long rows (thus giving the appearance of 4 \"columns\") \n thus we render firstCells of each category as its its own row, secondCells as own row, etc\n row 1 V&S T&O F&O D&M\n row 2 5 9 13 17\n row 3 6 10 14 18\n row 4 7 11 15 19\n row 5 8 12 16 20\n\n in this case, the answers table has 5 (1 row for pillars) rows that are really \"wide but short height wise\"\n */\n for (let i = 0; i < 4; i++) {\n const firstCellIndex = mobile ? i * 4 + 5 : i + 5;\n const cellsArray: Array = [<>>, <>>, <>>, <>>];\n for (let j = 0; j < 4; j++) {\n const currentFocusArea = mobile\n ? j + firstCellIndex\n : j * 4 + firstCellIndex;\n const userAnswerValue: number =\n parseInt(userAnswers[currentFocusArea].data.replace('\"', '')) - 1; // 0, 1, or 2\n const userFocusAreaLevel = focusAreaLevels[userAnswerValue];\n const focusAreaChipClass = focusAreaChipClasses[userAnswerValue];\n const cell = (\n \n \n \n {`${questions[currentFocusArea].title}`}\n \n \n {userFocusAreaLevel}\n \n \n \n {questions[currentFocusArea].options[userAnswerValue].title}\n \n \n );\n cellsArray[j + 1] = cell;\n }\n\n // process to add a row to render\n if (mobile) {\n cellsArray[0] = (\n \n \n {pillars[i]}\n \n \n );\n\n const row = (\n \n {cellsArray[0]}\n {cellsArray[1]}\n {cellsArray[2]}\n {cellsArray[3]}\n {cellsArray[4]}\n \n );\n tableItems[i] = row;\n } else {\n const row = (\n \n {cellsArray[1]}\n {cellsArray[2]}\n {cellsArray[3]}\n {cellsArray[4]}\n \n );\n tableItems[i + 1] = row;\n }\n }\n\n if (!mobile) {\n const pillarsArray = [<>>, <>>, <>>, <>>];\n for (let i = 0; i < 4; i++) {\n pillarsArray[i] = (\n \n \n {pillars[i]}\n \n \n );\n }\n const pillarsRow = (\n \n {pillarsArray[0]}\n {pillarsArray[1]}\n {pillarsArray[2]}\n {pillarsArray[3]}\n \n );\n tableItems[0] = pillarsRow;\n }\n\n return tableItems;\n};\n\nexport const generateTableLegend = (descriptions: string[]) => {\n const colors = [\n {\n color: 'rgba(146, 0, 0, 1)',\n backgroundColor: 'rgba(255, 204, 204, 1)',\n },\n {\n color: 'rgba(255, 152, 0, 1)',\n backgroundColor: 'rgba(250, 243, 185, 1)',\n },\n {\n color: 'rgba(0, 111, 59, 1)',\n backgroundColor: 'rgba(232, 249, 241, 1)',\n },\n ];\n const scoring = [\n <>\n \n Lagging\n \n \n >,\n <>\n \n Sustainable\n \n \n >,\n <>\n \n Leading\n \n \n >,\n ];\n const urgencies = [\n <>\n Requires immediate attention\n >,\n 'Needs to be addressed in the near-term',\n 'Continue to improve in long-term transformations',\n ];\n const effectivenesses = [\n 'Characterized by inefficacy',\n 'Organization is functioning, but not at an optimal level',\n 'Displaying operational excellence',\n ];\n const scalabilities = [\n <>\n Completely unscalable. Scale risks foundational failure\n >,\n 'Difficult to scale. Scale would further expose weaknesses and add pressure on teams',\n 'Scalable with few concerns ',\n ];\n\n const scoringCells = [<>>, <>>, <>>];\n for (let i = 0; i < 3; i++) {\n const cell = (\n \n \n {scoring[i]}\n \n \n \n {descriptions[i]}\n \n \n URGENCY: \n {urgencies[i]}\n \n \n \n OPERATIONAL EFFECTIVENESS:{' '}\n \n {effectivenesses[i]}\n \n \n SCALABILITY: \n {scalabilities[i]}\n \n \n \n );\n scoringCells[i] = cell;\n }\n\n return (\n \n \n {scoringCells[0]}\n \n \n \n {scoringCells[1]}\n \n \n \n {scoringCells[2]}\n \n \n );\n};\n","import { FC } from 'react';\nimport { Typography, Grid } from '@mui/material';\nimport { FormPublicDesignFragment } from 'graphql/fragment/form.public.fragment';\nimport AndrewKlajbor from '../assets/andrewklajbor.jpg';\nimport AnthonyJardim from '../assets/anthonyjardim.jpg';\nimport MenesKum from '../assets/meneskum.jpg';\nimport VincentAttonito from '../assets/vincentattonito.jpg';\nimport Supporter from './Supporter';\n\ninterface Props {\n design: FormPublicDesignFragment;\n}\n\nconst ContactUs: FC = ({ design }) => {\n return (\n <>\n \n \n For more information, please contact us:\n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n >\n );\n};\n\nexport default ContactUs;\n","import { useState, useEffect, CSSProperties } from 'react';\nimport {\n Chart as ChartJS,\n RadialLinearScale,\n PointElement,\n LineElement,\n Filler,\n Tooltip,\n Legend,\n} from 'chart.js';\nimport { Radar } from 'react-chartjs-2';\nimport {\n Box,\n Grid,\n Typography,\n Link,\n Stack,\n Modal,\n CircularProgress,\n} from '@mui/material';\nimport { saveAs } from 'file-saver';\nimport { StyledButton } from '@/components/styled/button';\nimport { FormPublicFieldFragment } from 'graphql/fragment/form.public.fragment';\nimport MenuBookTwoToneIcon from '@mui/icons-material/MenuBookTwoTone';\nimport OpenInNewIcon from '@mui/icons-material/OpenInNew';\nimport { lighten } from 'polished';\nimport html2canvas from 'html2canvas';\nimport ContactUs from './components/ContactUs';\nimport {\n generateAnswersTable,\n generateTableLegend,\n} from './components/ResultsTable';\n\ninterface UserAnswer {\n field: string; // question id\n data: string; // user chosen question answer\n}\n\nconst PillarsOfProductShape = ({\n scores,\n design,\n jsonData,\n questions,\n userAnswers,\n}: any) => {\n ChartJS.register(\n RadialLinearScale,\n PointElement,\n LineElement,\n Filler,\n Tooltip,\n Legend,\n {\n id: 'custom_canvas_background_color',\n beforeDraw: (chart) => {\n const ctx = chart.canvas.getContext('2d');\n ctx!.save();\n ctx!.globalCompositeOperation = 'destination-over';\n ctx!.fillStyle = '#000000';\n ctx!.fillRect(0, 0, chart.width, chart.height);\n ctx!.restore();\n },\n }\n );\n\n const [saving, setSaving] = useState(false);\n\n const saveResults = async () => {\n const endPage = document.getElementById('endpage-div');\n const canvas = await html2canvas(endPage!);\n canvas.toBlob(function (blob) {\n saveAs(blob!, 'pillars-of-product-chart.png');\n });\n };\n\n // used to show saving modal for a second, without it modal just flashes in and out\n const stopSaving = async () => {\n const sleep = (ms: number) => new Promise((r) => setTimeout(r, ms));\n await sleep(1000);\n setSaving(false);\n };\n\n useEffect(() => {\n if (saving) {\n saveResults();\n stopSaving();\n }\n }, [saving]);\n\n const modalStyling: CSSProperties = {\n position: 'absolute' as 'absolute',\n top: '50%',\n left: '50%',\n transform: 'translate(-50%, -50%)',\n border: `2px solid ${design.colors.question}`,\n borderRadius: '8px',\n backgroundColor: design.colors.background,\n color: design.colors.button,\n padding: '40px',\n };\n\n const saveChart = () => {\n //save to png\n const canvasSave = document.getElementById('pillars-of-product-chart');\n // @ts-ignore: Object is possibly 'null'.\n canvasSave.toBlob(function (blob) {\n saveAs(blob, 'pillars-of-product-chart.png');\n });\n };\n\n // console.log(chart.config.options) to see this object\n const chartData = {\n labels: jsonData.chartCategories,\n datasets: [\n {\n label: 'Shape',\n data: scores,\n backgroundColor: 'rgba(134, 188, 37, 0.2)',\n borderColor: 'rgba(134, 188, 37, 1)',\n color: 'rgba(255, 255, 255, 0.75)',\n borderWidth: 1,\n },\n ],\n };\n\n // console.log(chart.config.options) to see this object\n const chartOptions: {\n plugins: {\n legend: {\n display: boolean;\n };\n };\n scales: {\n r: {\n min: number;\n max: number;\n ticks: {\n stepSize: number;\n callback: any;\n };\n grid: {\n color: string;\n lineWidth: number;\n };\n pointLabels: any;\n };\n };\n } = {\n plugins: {\n legend: {\n display: false,\n },\n },\n scales: {\n r: {\n min: 0,\n max: 3,\n ticks: {\n stepSize: 1,\n callback: function (index: number): string {\n switch (index) {\n case 1:\n return 'Lagging';\n case 2:\n return 'Sustainable';\n case 3:\n return 'Leading';\n default:\n return '';\n }\n },\n },\n grid: {\n color: 'rgba(255, 255, 255, 0.75)',\n lineWidth: 2,\n },\n pointLabels: {\n /* Adjust data label font size according to chart size */\n font: function (context: { chart: { width: any } }) {\n var width = context.chart.width;\n var size = Math.round(width / 48);\n return {\n size: size,\n };\n },\n color: 'rgba(255, 255, 255, 1)',\n },\n },\n },\n };\n\n if (userAnswers.length !== 21) {\n return (\n
\n Something is wrong with the submission. The amount of answers don't\n match the amount of questions.\n
\n );\n }\n\n return (\n
\n \n \n \n \n \n
\n \n \n \n Thank you for completing your Product Capability Assessment,{' '}\n {userAnswers[0].data.replace(/\\\"/g, '')}.\n \n \n \n \n Thank you for taking the time to complete our survey on the 4\n pillars of product-centric organizational design. Your inputs have\n been scored across Deloitte's 16 dimensions of product centricity\n and ranked as Lagging, Leading, or Sustaining.\n {'\\n\\n'}This scoring is focused on delivering exceptional products\n and services to your customers. We appreciate your willingness to\n share your insights and experiences with us, and to learn more\n about Deloitte's 4 Pillars of Product Management please read our\n white paper{' '}\n {saving ? null : (\n \n here\n \n \n )}\n {'.'}\n \n \n \n \n Product Capability Assessment Details\n \n \n {generateAnswersTable(\n questions,\n [\n 'Vision &\\nStrategy',\n 'Talent &\\nOrganization',\n 'Focus on Users &\\nOutcomes',\n 'Delivery &\\nMindset',\n ],\n userAnswers,\n window.innerWidth <= 900\n )}\n \n \n Scoring is defined as follows:\n \n \n \n {generateTableLegend([\n 'Current operations fall below outlined standards compared to market competitors and require material structural adjustments.',\n 'Current operations fall below outlined standards compared to market competitors and require material structural adjustments.',\n 'Current operations exceed outlined standards compared to market competitors and differentiate from competition.',\n ])}\n \n \n \n Your next steps\n \n \n \n \n \n \n \n Now that you have shared your insights and experiences with\n us, we invite you to take the next step and join us in our\n efforts to create more product-centric organizations. Please\n reach out to learn more about how Deloitte is leading\n organizational changes to become more product-centric.\n \n \n \n \n {saving ? null : (\n \n \n Contact\n \n setSaving(true)}\n style={{\n fontWeight: 'bold',\n width: '75%',\n }}\n disableRipple\n >\n SAVE YOUR RESULTS\n \n \n )}\n \n \n \n \n \n \n \n \n \n \n \n \n
\n
\n );\n};\n\nexport default PillarsOfProductShape;\n","import { useEffect, useRef, useState } from 'react';\n\ninterface Props {\n design?: any;\n url: string;\n}\n\nconst determineImage = (hexCode: string) => {\n // old color data sometimes have 3 hexcode string for black or white\n // may need to remove (deprecate) this if block once things settle\n if (hexCode === 'FFF' || hexCode === 'FFFFFF') {\n return require('../assets/images/form-start-image-black.png');\n } else if (hexCode === '000' || hexCode === '000000') {\n return require('../assets/images/form-start-image-white.png');\n }\n\n // https://stackoverflow.com/questions/3942878/how-to-decide-font-color-in-white-or-black-depending-on-background-color\n const red = parseInt(hexCode.substring(0, 2), 16);\n const green = parseInt(hexCode.substring(2, 4), 16);\n const blue = parseInt(hexCode.substring(4, 6), 16);\n return red * 0.299 + green * 0.587 + blue * 0.114 > 186\n ? require('../assets/images/form-start-image-black.png')\n : require('../assets/images/form-start-image-white.png');\n};\n\nexport const StartImage: React.FC = (props) => {\n const [imageClass, setImageClass] = useState(\n 'form-start-page-image-landscape'\n );\n const ref: any = useRef(null);\n useEffect(() => {\n const width = ref.current!.naturalWidth;\n const height = ref.current!.naturalHeight;\n const ratio = height / width;\n if (ratio >= 0.8) {\n setImageClass('form-start-page-image-portrait');\n }\n }, []);\n\n const imageURL =\n props.url === ''\n ? determineImage(\n props.design.colors.background.toUpperCase().substring(1) // substring(1) to get rid of pound sign in hexcode\n )\n : props.url;\n\n return (\n