Use high quality render for CadQuery download

This commit is contained in:
Kurt Hutten
2021-07-03 13:07:32 +10:00
parent 70e55a039d
commit f176bbe090
12 changed files with 144 additions and 58 deletions

View File

@@ -37,6 +37,8 @@ yarn rw prisma migrate dev
yarn rw prisma db seed yarn rw prisma db seed
``` ```
p.s. `yarn rw prisma studio` spins up an app to inspect the db
### Fire up dev ### Fire up dev
```terminal ```terminal
yarn rw dev yarn rw dev

View File

@@ -22,7 +22,7 @@ CLOUDINARY_API_KEY=476712943135152
# EMAIL_PASSWORD=abc123 # EMAIL_PASSWORD=abc123
CAD_LAMBDA_BASE_URL="http://localhost:8080" # CAD_LAMBDA_BASE_URL="http://localhost:8080"
# sentry # sentry
GITHUB_ASSIST_APP_ID=23342 GITHUB_ASSIST_APP_ID=23342

View File

@@ -10,7 +10,9 @@ const EditorMenu = () => {
const { state, thunkDispatch } = useIdeContext() const { state, thunkDispatch } = useIdeContext()
const handleStlDownload = makeStlDownloadHandler({ const handleStlDownload = makeStlDownloadHandler({
type: state.objectData?.type, type: state.objectData?.type,
ideType: state.ideType,
geometry: state.objectData?.data, geometry: state.objectData?.data,
quality: state.objectData?.quality,
fileName: PullTitleFromFirstLine(state.code || ''), fileName: PullTitleFromFirstLine(state.code || ''),
thunkDispatch, thunkDispatch,
}) })

View File

@@ -2,7 +2,7 @@ import { flow, identity } from 'lodash/fp'
import { fileSave } from 'browser-fs-access' import { fileSave } from 'browser-fs-access'
import { MeshBasicMaterial, Mesh, Scene } from 'three' import { MeshBasicMaterial, Mesh, Scene } from 'three'
import { STLExporter } from 'three/examples/jsm/exporters/STLExporter' import { STLExporter } from 'three/examples/jsm/exporters/STLExporter'
import { requestRender } from 'src/helpers/hooks/useIdeState' import { requestRender, State } from 'src/helpers/hooks/useIdeState'
export const PullTitleFromFirstLine = (code = '') => { export const PullTitleFromFirstLine = (code = '') => {
const firstLine = code.split('\n').filter(identity)[0] || '' const firstLine = code.split('\n').filter(identity)[0] || ''
@@ -16,8 +16,24 @@ export const PullTitleFromFirstLine = (code = '') => {
) )
} }
interface makeStlDownloadHandlerArgs {
geometry: any
fileName: string
type: State['objectData']['type']
ideType: State['ideType']
thunkDispatch: (a: any) => any
quality: State['objectData']['quality']
}
export const makeStlDownloadHandler = export const makeStlDownloadHandler =
({ geometry, fileName, type, thunkDispatch }) => ({
geometry,
fileName,
type,
thunkDispatch,
quality,
ideType,
}: makeStlDownloadHandlerArgs) =>
() => { () => {
const makeStlBlobFromGeo = flow( const makeStlBlobFromGeo = flow(
(geo) => new Mesh(geo, new MeshBasicMaterial()), (geo) => new Mesh(geo, new MeshBasicMaterial()),
@@ -36,22 +52,25 @@ export const makeStlDownloadHandler =
}) })
} }
if (geometry) { if (geometry) {
if (type === 'geometry') { if (
type === 'geometry' &&
(quality === 'high' || ideType === 'openScad')
) {
saveFile(geometry) saveFile(geometry)
} else { } else {
thunkDispatch((dispatch, getState) => { thunkDispatch((dispatch, getState) => {
const state = getState() const state = getState()
if (state.ideType === 'openScad') { const specialCadProcess = ideType === 'openScad' && 'stl'
dispatch({ type: 'setLoading' }) dispatch({ type: 'setLoading' })
requestRender({ requestRender({
state, state,
dispatch, dispatch,
code: state.code, code: state.code,
viewerSize: state.viewerSize, viewerSize: state.viewerSize,
camera: state.camera, camera: state.camera,
specialCadProcess: 'stl', quality: 'high',
}).then((result) => result && saveFile(result.data)) specialCadProcess,
} }).then((result) => result && saveFile(result.data))
}) })
} }
} }

View File

@@ -1,7 +1,6 @@
import { useEffect, useState } from 'react' import { useEffect, useState } from 'react'
import { useIdeContext } from 'src/helpers/hooks/useIdeContext' import { useIdeContext } from 'src/helpers/hooks/useIdeContext'
import { makeCodeStoreKey } from 'src/helpers/hooks/useIdeState' import { makeCodeStoreKey, requestRender } from 'src/helpers/hooks/useIdeState'
import { requestRender } from 'src/helpers/hooks/useIdeState'
import Editor, { useMonaco } from '@monaco-editor/react' import Editor, { useMonaco } from '@monaco-editor/react'
import { theme } from 'src/../tailwind.config' import { theme } from 'src/../tailwind.config'

View File

@@ -1,5 +1,4 @@
import { makeCodeStoreKey } from 'src/helpers/hooks/useIdeState' import { makeCodeStoreKey, requestRender } from 'src/helpers/hooks/useIdeState'
import { requestRender } from 'src/helpers/hooks/useIdeState'
import { useIdeContext } from 'src/helpers/hooks/useIdeContext' import { useIdeContext } from 'src/helpers/hooks/useIdeContext'
export const useRender = () => { export const useRender = () => {

View File

@@ -4,12 +4,16 @@ import {
createHealthyResponse, createHealthyResponse,
createUnhealthyResponse, createUnhealthyResponse,
timeoutErrorMessage, timeoutErrorMessage,
RenderArgs,
} from './common' } from './common'
export const render = async ({ code }) => { export const render = async ({
code,
settings: { quality = 'low' },
}: RenderArgs) => {
const body = JSON.stringify({ const body = JSON.stringify({
settings: { settings: {
deflection: 0.15, deflection: quality === 'low' ? 0.35 : 0.11,
}, },
file: code, file: code,
}) })

View File

@@ -1,14 +1,24 @@
import { STLLoader } from 'three/examples/jsm/loaders/STLLoader' import { STLLoader } from 'three/examples/jsm/loaders/STLLoader'
import { State } from 'src/helpers/hooks/useIdeState'
export const lambdaBaseURL = export const lambdaBaseURL =
// process.env.CAD_LAMBDA_BASE_URL || process.env.CAD_LAMBDA_BASE_URL ||
'https://oxt2p7ddgj.execute-api.us-east-1.amazonaws.com/prod' 'https://2inlbple1b.execute-api.us-east-1.amazonaws.com/prod2'
export const stlToGeometry = (url) => export const stlToGeometry = (url) =>
new Promise((resolve, reject) => { new Promise((resolve, reject) => {
new STLLoader().load(url, resolve, null, reject) new STLLoader().load(url, resolve, null, reject)
}) })
export interface RenderArgs {
code: State['code']
settings: {
camera: State['camera']
viewerSize: State['viewerSize']
quality: State['objectData']['quality']
}
}
export function createHealthyResponse({ date, data, consoleMessage, type }) { export function createHealthyResponse({ date, data, consoleMessage, type }) {
return { return {
status: 'healthy', status: 'healthy',

View File

@@ -4,9 +4,10 @@ import {
createHealthyResponse, createHealthyResponse,
createUnhealthyResponse, createUnhealthyResponse,
timeoutErrorMessage, timeoutErrorMessage,
RenderArgs,
} from './common' } from './common'
export const render = async ({ code, settings }) => { export const render = async ({ code, settings }: RenderArgs) => {
const pixelRatio = window.devicePixelRatio || 1 const pixelRatio = window.devicePixelRatio || 1
const size = { const size = {
x: Math.round(settings.viewerSize?.width * pixelRatio), x: Math.round(settings.viewerSize?.width * pixelRatio),
@@ -67,7 +68,7 @@ export const render = async ({ code, settings }) => {
} }
} }
export const stl = async ({ code, settings }) => { export const stl = async ({ code, settings }: RenderArgs) => {
const body = JSON.stringify({ const body = JSON.stringify({
settings: {}, settings: {},
file: code, file: code,

View File

@@ -43,36 +43,64 @@ show_object(result)
const codeStorageKey = 'Last-editor-code' const codeStorageKey = 'Last-editor-code'
export const makeCodeStoreKey = (ideType) => `${codeStorageKey}-${ideType}` export const makeCodeStoreKey = (ideType) => `${codeStorageKey}-${ideType}`
let mutableState = null let mutableState: State = null
export const useIdeState = () => { interface XYZ {
const code = '' x: number
const initialLayout = { y: number
direction: 'row', z: number
first: 'Editor', }
second: {
direction: 'column', export interface State {
first: 'Viewer', ideType: 'INIT' | 'openScad' | 'cadQuery'
second: 'Console', consoleMessages: { type: 'message' | 'error'; message: string; time: Date }[]
splitPercentage: 70, code: string
}, objectData: {
type: 'INIT' | 'stl' | 'png' | 'geometry'
data: any
quality: 'low' | 'high'
} }
const initialState = { layout: any
ideType: 'INIT', camera: {
consoleMessages: [ dist?: number
{ type: 'message', message: 'Initialising', time: new Date() }, position?: XYZ
], rotation?: XYZ
code,
objectData: {
type: 'INIT',
data: null,
},
layout: initialLayout,
camera: {},
viewerSize: { width: 0, height: 0 },
isLoading: false,
} }
const reducer = (state, { type, payload }) => { viewerSize: { width: number; height: number }
isLoading: boolean
}
const code = ''
const initialLayout = {
direction: 'row',
first: 'Editor',
second: {
direction: 'column',
first: 'Viewer',
second: 'Console',
splitPercentage: 70,
},
}
export const initialState: State = {
ideType: 'INIT',
consoleMessages: [
{ type: 'message', message: 'Initialising', time: new Date() },
],
code,
objectData: {
type: 'INIT',
data: null,
quality: 'low',
},
layout: initialLayout,
camera: {},
viewerSize: { width: 0, height: 0 },
isLoading: false,
}
export const useIdeState = (): [State, (actionOrThunk: any) => any] => {
const reducer = (state: State, { type, payload }): State => {
switch (type) { switch (type) {
case 'initIde': case 'initIde':
return { return {
@@ -89,6 +117,7 @@ export const useIdeState = () => {
return { return {
...state, ...state,
objectData: { objectData: {
...state.objectData,
type: payload.objectData?.type, type: payload.objectData?.type,
data: payload.objectData?.data, data: payload.objectData?.data,
}, },
@@ -142,8 +171,19 @@ export const useIdeState = () => {
const [state, dispatch] = useReducer(reducer, initialState) const [state, dispatch] = useReducer(reducer, initialState)
mutableState = state mutableState = state
const getState = () => mutableState const getState = (): State => mutableState
return [state, withThunk(dispatch, getState)] const thunkDispatch = withThunk(dispatch, getState)
return [state, thunkDispatch]
}
interface RequestRenderArgs {
state: State
dispatch: any
code: State['code']
camera: State['camera']
viewerSize: State['viewerSize']
quality: State['objectData']['quality']
specialCadProcess?: string
} }
export const requestRender = ({ export const requestRender = ({
@@ -152,8 +192,9 @@ export const requestRender = ({
code, code,
camera, camera,
viewerSize, viewerSize,
quality,
specialCadProcess = null, specialCadProcess = null,
}) => { }: RequestRenderArgs) => {
if ( if (
state.ideType !== 'INIT' && state.ideType !== 'INIT' &&
(!state.isLoading || state.objectData?.type === 'INIT') (!state.isLoading || state.objectData?.type === 'INIT')
@@ -166,6 +207,7 @@ export const requestRender = ({
settings: { settings: {
camera, camera,
viewerSize, viewerSize,
quality,
}, },
}) })
.then(({ objectData, message, status }) => { .then(({ objectData, message, status }) => {

View File

@@ -1,10 +1,18 @@
import { createContext } from 'react' import { createContext } from 'react'
import Seo from 'src/components/Seo/Seo' import Seo from 'src/components/Seo/Seo'
import IdeWrapper from 'src/components/IdeWrapper' import IdeWrapper from 'src/components/IdeWrapper/IdeWrapper'
import { Toaster } from '@redwoodjs/web/toast' import { Toaster } from '@redwoodjs/web/toast'
import { useIdeState } from 'src/helpers/hooks/useIdeState' import { useIdeState, State, initialState } from 'src/helpers/hooks/useIdeState'
export const IdeContext = createContext() interface IdeContextType {
state: State
thunkDispatch: (actionOrThunk: any) => any
}
export const IdeContext = createContext<IdeContextType>({
state: initialState,
thunkDispatch: () => {},
})
const DevIdePage = ({ cadPackage }) => { const DevIdePage = ({ cadPackage }) => {
const [state, thunkDispatch] = useIdeState() const [state, thunkDispatch] = useIdeState()
return ( return (