sonamu 0.7.40 → 0.7.42

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.
@@ -1 +1 @@
1
- {"version":3,"file":"generated_http.template.d.ts","sourceRoot":"","sources":["../../../src/template/implementations/generated_http.template.ts"],"names":[],"mappings":"AACA,OAAO,EAAE,CAAC,EAAE,MAAM,KAAK,CAAC;AAExB,OAAO,KAAK,EAAE,WAAW,EAAE,MAAM,sBAAsB,CAAC;AAExD,OAAO,EAAE,QAAQ,EAAE,MAAM,aAAa,CAAC;AAEvC,qBAAa,wBAAyB,SAAQ,QAAQ;;IAKpD,gBAAgB;;;;IASV,MAAM;;;;;;IA8CZ,mBAAmB,CAAC,OAAO,EAAE,CAAC,CAAC,OAAO,CAAC,OAAO,CAAC,EAAE,IAAI,EAAE,MAAM,GAAG,OAAO;IAgEvE,gBAAgB,CACd,GAAG,EAAE,WAAW,EAEhB,UAAU,EAAE;QAAE,CAAC,QAAQ,EAAE,MAAM,GAAG,CAAC,CAAC,SAAS,CAAC,GAAG,CAAC,CAAA;KAAE,GACnD;QAAE,CAAC,GAAG,EAAE,MAAM,GAAG,OAAO,CAAA;KAAE;CAY9B"}
1
+ {"version":3,"file":"generated_http.template.d.ts","sourceRoot":"","sources":["../../../src/template/implementations/generated_http.template.ts"],"names":[],"mappings":"AACA,OAAO,EAAE,CAAC,EAAE,MAAM,KAAK,CAAC;AAExB,OAAO,KAAK,EAAE,WAAW,EAAE,MAAM,sBAAsB,CAAC;AAExD,OAAO,EAAE,QAAQ,EAAE,MAAM,aAAa,CAAC;AAEvC,qBAAa,wBAAyB,SAAQ,QAAQ;;IAKpD,gBAAgB;;;;IASV,MAAM;;;;;;IA8CZ,mBAAmB,CAAC,OAAO,EAAE,CAAC,CAAC,OAAO,CAAC,OAAO,CAAC,EAAE,IAAI,EAAE,MAAM,GAAG,OAAO;IAiEvE,gBAAgB,CACd,GAAG,EAAE,WAAW,EAEhB,UAAU,EAAE;QAAE,CAAC,QAAQ,EAAE,MAAM,GAAG,CAAC,CAAC,SAAS,CAAC,GAAG,CAAC,CAAA;KAAE,GACnD;QAAE,CAAC,GAAG,EAAE,MAAM,GAAG,OAAO,CAAA;KAAE;CAY9B"}
@@ -63,7 +63,8 @@ export class Template__generated_http extends Template {
63
63
  return [
64
64
  this.zodTypeToReqDefault(zodType.element, name)
65
65
  ];
66
- } else if (zodType instanceof z.ZodString) {
66
+ } else if (zodType instanceof z.core.$ZodString) {
67
+ // NOTE: z.ZodString으로 비교하면 z.url(), z.email() 등의 타입에서 문제가 생기므로 z.core.$ZodString으로 비교함
67
68
  if (name.endsWith("_at") || name.endsWith("_date") || name === "range") {
68
69
  return "2000-01-01";
69
70
  } else {
@@ -128,4 +129,4 @@ export class Template__generated_http extends Template {
128
129
  }
129
130
  }
130
131
 
131
- //# sourceMappingURL=data:application/json;base64,{"version":3,"sources":["../../../src/template/implementations/generated_http.template.ts"],"sourcesContent":["import qs from \"qs\";\nimport { z } from \"zod\";\nimport { getZodObjectFromApi } from \"../../api/code-converters\";\nimport type { ExtendedApi } from \"../../api/decorators\";\nimport { Sonamu } from \"../../api/sonamu\";\nimport { Template } from \"../template\";\n\nexport class Template__generated_http extends Template {\n  constructor() {\n    super(\"generated_http\");\n  }\n\n  getTargetAndPath() {\n    const { dir } = Sonamu.config.api;\n\n    return {\n      target: `${dir}/src/application`,\n      path: `sonamu.generated.http`,\n    };\n  }\n\n  async render() {\n    const {\n      syncer: { types, apis },\n      config: {\n        api: {\n          route: { prefix },\n        },\n      },\n    } = Sonamu;\n\n    const lines = await Promise.all(\n      apis.map(async (api) => {\n        const reqObject = this.resolveApiParams(api, types);\n\n        const dataLines = await (async () => {\n          if ((api.options.httpMethod ?? \"GET\") === \"GET\") {\n            return {\n              querystring: [qs.stringify(reqObject, { encode: false }).split(\"&\").join(\"\\n\\t&\")],\n              body: [],\n            };\n          } else {\n            return {\n              querystring: [],\n              body: [\"\", JSON.stringify(reqObject, null, 2)],\n            };\n          }\n        })();\n\n        return [\n          [\n            `${api.options.httpMethod ?? \"GET\"} {{baseUrl}}${prefix}${api.path}`,\n            ...dataLines.querystring,\n          ].join(\"\\n\\t?\"),\n          `Content-Type: ${api.options.contentType ?? \"application/json\"}`,\n          ...dataLines.body,\n        ].join(\"\\n\");\n      }),\n    );\n\n    return {\n      ...this.getTargetAndPath(),\n      body: lines.join(\"\\n\\n###\\n\\n\"),\n      importKeys: [],\n    };\n  }\n\n  zodTypeToReqDefault(zodType: z.ZodType<unknown>, name: string): unknown {\n    if (zodType instanceof z.ZodObject) {\n      return Object.fromEntries(\n        Object.keys(zodType.shape).map((key) => [\n          key,\n          this.zodTypeToReqDefault(zodType.shape[key], key),\n        ]),\n      );\n    } else if (zodType instanceof z.ZodArray) {\n      return [this.zodTypeToReqDefault((zodType as z.ZodArray<z.ZodType>).element, name)];\n    } else if (zodType instanceof z.ZodString) {\n      if (name.endsWith(\"_at\") || name.endsWith(\"_date\") || name === \"range\") {\n        return \"2000-01-01\";\n      } else {\n        return name.toUpperCase();\n      }\n    } else if (zodType instanceof z.ZodNumber) {\n      if (name === \"num\") {\n        return 24;\n      }\n\n      const minValue = zodType.minValue ?? 0;\n      return minValue > Number.MIN_SAFE_INTEGER ? minValue : 0;\n    } else if (zodType instanceof z.ZodBoolean) {\n      return false;\n    } else if (zodType instanceof z.ZodEnum) {\n      return zodType.options[0];\n    } else if (zodType instanceof z.ZodOptional) {\n      return this.zodTypeToReqDefault((zodType as z.ZodOptional<z.ZodType>).def.innerType, name);\n    } else if (zodType instanceof z.ZodNullable) {\n      return null;\n    } else if (zodType instanceof z.ZodUnion) {\n      return this.zodTypeToReqDefault((zodType as z.ZodUnion<z.ZodType[]>).def.options[0], name);\n    } else if (zodType instanceof z.ZodUnknown) {\n      return \"unknown\";\n    } else if (zodType instanceof z.ZodTuple) {\n      /** biome-ignore lint/suspicious/noExplicitAny: ZodTuple 타입 사용 */\n      return zodType.def.items.map((item: any) => this.zodTypeToReqDefault(item, name));\n    } else if (zodType instanceof z.ZodDate) {\n      return \"2000-01-01\";\n    } else if (zodType instanceof z.ZodLiteral) {\n      return zodType.value;\n    } else if (zodType instanceof z.ZodRecord || zodType instanceof z.ZodMap) {\n      // biome-ignore lint/suspicious/noExplicitAny: ZodRecord 타입 사용\n      const kvDef = (zodType as z.ZodRecord<any, z.ZodType> | z.ZodMap<z.ZodType, z.ZodType>).def;\n      // biome-ignore lint/suspicious/noExplicitAny: ZodIntersection 타입 사용\n      const key = this.zodTypeToReqDefault(kvDef.keyType, name) as any;\n      const value = this.zodTypeToReqDefault(kvDef.valueType, name);\n      return { [key]: value };\n    } else if (zodType instanceof z.ZodSet) {\n      return [this.zodTypeToReqDefault((zodType as z.ZodSet<z.ZodType>).def.valueType, name)];\n    } else if (zodType instanceof z.ZodIntersection) {\n      return this.zodTypeToReqDefault(\n        (zodType as z.ZodIntersection<z.ZodType, z.ZodType>).def.right,\n        name,\n      );\n    } else if (zodType instanceof z.ZodDefault) {\n      return this.zodTypeToReqDefault((zodType as z.ZodDefault<z.ZodType>).def.innerType, name);\n    } else {\n      // console.log(zodType);\n      return `unknown-${zodType.type}`;\n    }\n  }\n\n  resolveApiParams(\n    api: ExtendedApi,\n    // biome-ignore lint/suspicious/noExplicitAny: ZodObject 타입 사용\n    references: { [typeName: string]: z.ZodObject<any> },\n  ): { [key: string]: unknown } {\n    const reqType = getZodObjectFromApi(api, references);\n    try {\n      const def = this.zodTypeToReqDefault(reqType, \"unknownName\") as {\n        [key: string]: unknown;\n      };\n      return def;\n    } catch (error) {\n      console.error(error);\n      throw new Error(`Invalid zod type detected on ${api.modelName}:${api.methodName}`);\n    }\n  }\n}\n"],"names":["qs","z","getZodObjectFromApi","Sonamu","Template","Template__generated_http","getTargetAndPath","dir","config","api","target","path","render","syncer","types","apis","route","prefix","lines","Promise","all","map","reqObject","resolveApiParams","dataLines","options","httpMethod","querystring","stringify","encode","split","join","body","JSON","contentType","importKeys","zodTypeToReqDefault","zodType","name","ZodObject","Object","fromEntries","keys","shape","key","ZodArray","element","ZodString","endsWith","toUpperCase","ZodNumber","minValue","Number","MIN_SAFE_INTEGER","ZodBoolean","ZodEnum","ZodOptional","def","innerType","ZodNullable","ZodUnion","ZodUnknown","ZodTuple","items","item","ZodDate","ZodLiteral","value","ZodRecord","ZodMap","kvDef","keyType","valueType","ZodSet","ZodIntersection","right","ZodDefault","type","references","reqType","error","console","Error","modelName","methodName"],"mappings":"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"}
132
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"sources":["../../../src/template/implementations/generated_http.template.ts"],"sourcesContent":["import qs from \"qs\";\nimport { z } from \"zod\";\nimport { getZodObjectFromApi } from \"../../api/code-converters\";\nimport type { ExtendedApi } from \"../../api/decorators\";\nimport { Sonamu } from \"../../api/sonamu\";\nimport { Template } from \"../template\";\n\nexport class Template__generated_http extends Template {\n  constructor() {\n    super(\"generated_http\");\n  }\n\n  getTargetAndPath() {\n    const { dir } = Sonamu.config.api;\n\n    return {\n      target: `${dir}/src/application`,\n      path: `sonamu.generated.http`,\n    };\n  }\n\n  async render() {\n    const {\n      syncer: { types, apis },\n      config: {\n        api: {\n          route: { prefix },\n        },\n      },\n    } = Sonamu;\n\n    const lines = await Promise.all(\n      apis.map(async (api) => {\n        const reqObject = this.resolveApiParams(api, types);\n\n        const dataLines = await (async () => {\n          if ((api.options.httpMethod ?? \"GET\") === \"GET\") {\n            return {\n              querystring: [qs.stringify(reqObject, { encode: false }).split(\"&\").join(\"\\n\\t&\")],\n              body: [],\n            };\n          } else {\n            return {\n              querystring: [],\n              body: [\"\", JSON.stringify(reqObject, null, 2)],\n            };\n          }\n        })();\n\n        return [\n          [\n            `${api.options.httpMethod ?? \"GET\"} {{baseUrl}}${prefix}${api.path}`,\n            ...dataLines.querystring,\n          ].join(\"\\n\\t?\"),\n          `Content-Type: ${api.options.contentType ?? \"application/json\"}`,\n          ...dataLines.body,\n        ].join(\"\\n\");\n      }),\n    );\n\n    return {\n      ...this.getTargetAndPath(),\n      body: lines.join(\"\\n\\n###\\n\\n\"),\n      importKeys: [],\n    };\n  }\n\n  zodTypeToReqDefault(zodType: z.ZodType<unknown>, name: string): unknown {\n    if (zodType instanceof z.ZodObject) {\n      return Object.fromEntries(\n        Object.keys(zodType.shape).map((key) => [\n          key,\n          this.zodTypeToReqDefault(zodType.shape[key], key),\n        ]),\n      );\n    } else if (zodType instanceof z.ZodArray) {\n      return [this.zodTypeToReqDefault((zodType as z.ZodArray<z.ZodType>).element, name)];\n    } else if (zodType instanceof z.core.$ZodString) {\n      // NOTE: z.ZodString으로 비교하면 z.url(), z.email() 등의 타입에서 문제가 생기므로 z.core.$ZodString으로 비교함\n      if (name.endsWith(\"_at\") || name.endsWith(\"_date\") || name === \"range\") {\n        return \"2000-01-01\";\n      } else {\n        return name.toUpperCase();\n      }\n    } else if (zodType instanceof z.ZodNumber) {\n      if (name === \"num\") {\n        return 24;\n      }\n\n      const minValue = zodType.minValue ?? 0;\n      return minValue > Number.MIN_SAFE_INTEGER ? minValue : 0;\n    } else if (zodType instanceof z.ZodBoolean) {\n      return false;\n    } else if (zodType instanceof z.ZodEnum) {\n      return zodType.options[0];\n    } else if (zodType instanceof z.ZodOptional) {\n      return this.zodTypeToReqDefault((zodType as z.ZodOptional<z.ZodType>).def.innerType, name);\n    } else if (zodType instanceof z.ZodNullable) {\n      return null;\n    } else if (zodType instanceof z.ZodUnion) {\n      return this.zodTypeToReqDefault((zodType as z.ZodUnion<z.ZodType[]>).def.options[0], name);\n    } else if (zodType instanceof z.ZodUnknown) {\n      return \"unknown\";\n    } else if (zodType instanceof z.ZodTuple) {\n      /** biome-ignore lint/suspicious/noExplicitAny: ZodTuple 타입 사용 */\n      return zodType.def.items.map((item: any) => this.zodTypeToReqDefault(item, name));\n    } else if (zodType instanceof z.ZodDate) {\n      return \"2000-01-01\";\n    } else if (zodType instanceof z.ZodLiteral) {\n      return zodType.value;\n    } else if (zodType instanceof z.ZodRecord || zodType instanceof z.ZodMap) {\n      // biome-ignore lint/suspicious/noExplicitAny: ZodRecord 타입 사용\n      const kvDef = (zodType as z.ZodRecord<any, z.ZodType> | z.ZodMap<z.ZodType, z.ZodType>).def;\n      // biome-ignore lint/suspicious/noExplicitAny: ZodIntersection 타입 사용\n      const key = this.zodTypeToReqDefault(kvDef.keyType, name) as any;\n      const value = this.zodTypeToReqDefault(kvDef.valueType, name);\n      return { [key]: value };\n    } else if (zodType instanceof z.ZodSet) {\n      return [this.zodTypeToReqDefault((zodType as z.ZodSet<z.ZodType>).def.valueType, name)];\n    } else if (zodType instanceof z.ZodIntersection) {\n      return this.zodTypeToReqDefault(\n        (zodType as z.ZodIntersection<z.ZodType, z.ZodType>).def.right,\n        name,\n      );\n    } else if (zodType instanceof z.ZodDefault) {\n      return this.zodTypeToReqDefault((zodType as z.ZodDefault<z.ZodType>).def.innerType, name);\n    } else {\n      // console.log(zodType);\n      return `unknown-${zodType.type}`;\n    }\n  }\n\n  resolveApiParams(\n    api: ExtendedApi,\n    // biome-ignore lint/suspicious/noExplicitAny: ZodObject 타입 사용\n    references: { [typeName: string]: z.ZodObject<any> },\n  ): { [key: string]: unknown } {\n    const reqType = getZodObjectFromApi(api, references);\n    try {\n      const def = this.zodTypeToReqDefault(reqType, \"unknownName\") as {\n        [key: string]: unknown;\n      };\n      return def;\n    } catch (error) {\n      console.error(error);\n      throw new Error(`Invalid zod type detected on ${api.modelName}:${api.methodName}`);\n    }\n  }\n}\n"],"names":["qs","z","getZodObjectFromApi","Sonamu","Template","Template__generated_http","getTargetAndPath","dir","config","api","target","path","render","syncer","types","apis","route","prefix","lines","Promise","all","map","reqObject","resolveApiParams","dataLines","options","httpMethod","querystring","stringify","encode","split","join","body","JSON","contentType","importKeys","zodTypeToReqDefault","zodType","name","ZodObject","Object","fromEntries","keys","shape","key","ZodArray","element","core","$ZodString","endsWith","toUpperCase","ZodNumber","minValue","Number","MIN_SAFE_INTEGER","ZodBoolean","ZodEnum","ZodOptional","def","innerType","ZodNullable","ZodUnion","ZodUnknown","ZodTuple","items","item","ZodDate","ZodLiteral","value","ZodRecord","ZodMap","kvDef","keyType","valueType","ZodSet","ZodIntersection","right","ZodDefault","type","references","reqType","error","console","Error","modelName","methodName"],"mappings":"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"}
@@ -1 +1 @@
1
- {"version":3,"file":"queries.template.d.ts","sourceRoot":"","sources":["../../../src/template/implementations/queries.template.ts"],"names":[],"mappings":"AAKA,OAAO,KAAK,EAAE,eAAe,EAAE,MAAM,mBAAmB,CAAC;AAEzD,OAAO,EAAE,QAAQ,EAAE,MAAM,aAAa,CAAC;AAEvC,qBAAa,iBAAkB,SAAQ,QAAQ;;IAK7C,gBAAgB;;;;IAQhB,MAAM,CAAC,EAAE,EAAE,eAAe,CAAC,SAAS,CAAC;;;;;;;CAwFtC"}
1
+ {"version":3,"file":"queries.template.d.ts","sourceRoot":"","sources":["../../../src/template/implementations/queries.template.ts"],"names":[],"mappings":"AAKA,OAAO,KAAK,EAAE,eAAe,EAAE,MAAM,mBAAmB,CAAC;AAEzD,OAAO,EAAE,QAAQ,EAAE,MAAM,aAAa,CAAC;AAEvC,qBAAa,iBAAkB,SAAQ,QAAQ;;IAK7C,gBAAgB;;;;IAQhB,MAAM,CAAC,EAAE,EAAE,eAAe,CAAC,SAAS,CAAC;;;;;;;CAmGtC"}
@@ -60,11 +60,15 @@ ${functions.join("\n\n")}
60
60
  }
61
61
  `.trim());
62
62
  }
63
+ // tanstack-query API가 없으면 헬퍼 함수와 import를 포함하지 않습니다.
64
+ // 새 프로젝트에서 첫 빌드 시 sync가 아직 안 되어 namespace가 비어있을 수 있고,
65
+ // 이때 createSSRQuery가 unused로 빌드 에러가 발생하는 것을 방지합니다.
66
+ const hasQueries = namespaces.length > 0;
63
67
  return {
64
68
  ...this.getTargetAndPath(),
65
69
  body: namespaces.join("\n\n"),
66
70
  importKeys: diff(unique(importKeys), typeParamNames),
67
- customHeaders: [
71
+ customHeaders: hasQueries ? [
68
72
  "/** biome-ignore-all lint: generated는 무시 */",
69
73
  "/** biome-ignore-all assist: generated는 무시 */",
70
74
  "",
@@ -75,9 +79,13 @@ ${functions.join("\n\n")}
75
79
  ` return { modelName, methodName, params, serviceKey, __brand: 'SSRQuery' } as SSRQuery;`,
76
80
  `}`,
77
81
  ""
82
+ ] : [
83
+ "/** biome-ignore-all lint: generated는 무시 */",
84
+ "/** biome-ignore-all assist: generated는 무시 */",
85
+ ""
78
86
  ]
79
87
  };
80
88
  }
81
89
  }
82
90
 
83
- //# sourceMappingURL=data:application/json;base64,{"version":3,"sources":["../../../src/template/implementations/queries.template.ts"],"sourcesContent":["import inflection from \"inflection\";\nimport { diff, unique } from \"radashi\";\nimport { apiParamToTsCode, apiParamTypeToTsType } from \"../../api/code-converters\";\nimport type { ExtendedApi } from \"../../api/decorators\";\nimport { Sonamu } from \"../../api/sonamu\";\nimport type { TemplateOptions } from \"../../types/types\";\nimport { ApiParamType } from \"../../types/types\";\nimport { Template } from \"../template\";\n\nexport class Template__queries extends Template {\n  constructor() {\n    super(\"queries\");\n  }\n\n  getTargetAndPath() {\n    const { dir } = Sonamu.config.api;\n    return {\n      target: `${dir}/src/application`,\n      path: `queries.generated.ts`,\n    };\n  }\n\n  render({}: TemplateOptions[\"queries\"]) {\n    const { apis } = Sonamu.syncer;\n\n    // tanstack-query를 포함한 API만 필터링\n    const queryApis = apis.filter((api) => api.options.clients?.includes(\"tanstack-query\"));\n\n    // 모델별로 그룹화\n    const apisByModel = new Map<string, ExtendedApi[]>();\n    for (const api of queryApis) {\n      const modelName = api.modelName.replace(/Model$/, \"\").replace(/Frame$/, \"\");\n      if (!apisByModel.has(modelName)) {\n        apisByModel.set(modelName, []);\n      }\n      apisByModel.get(modelName)?.push(api);\n    }\n\n    const namespaces: string[] = [];\n    const importKeys: string[] = [];\n    let typeParamNames: string[] = [];\n\n    for (const [modelName, modelApis] of apisByModel) {\n      const functions: string[] = [];\n\n      for (const api of modelApis) {\n        // Context 제외한 파라미터\n        const paramsWithoutContext = api.parameters.filter(\n          (param) =>\n            !ApiParamType.isContext(param.type) &&\n            !ApiParamType.isRefKnex(param.type) &&\n            !(param.optional === true && param.name.startsWith(\"_\")),\n        );\n\n        // 타입 파라미터 이름 수집\n        typeParamNames = typeParamNames.concat(api.typeParameters.map((tp) => tp.id));\n\n        // 타입 파라미터 정의\n        const typeParametersAsTsType = api.typeParameters\n          .map((typeParam) => {\n            return apiParamTypeToTsType(typeParam, importKeys);\n          })\n          .join(\", \");\n        const typeParamsDef = typeParametersAsTsType ? `<${typeParametersAsTsType}>` : \"\";\n\n        // 파라미터 정의\n        const paramsDef = apiParamToTsCode(paramsWithoutContext, importKeys);\n        const paramNames = paramsWithoutContext.map((p) => p.name).join(\", \");\n\n        // serviceMethodName 계산 (services.template.ts와 동일한 로직)\n        const serviceMethodName = api.options.resourceName\n          ? `get${inflection.camelize(api.options.resourceName)}`\n          : api.methodName;\n\n        // SSRQuery 함수 생성 (함수명도 serviceMethodName 사용)\n        functions.push(\n          `\n  export const ${serviceMethodName} = ${typeParamsDef}(${paramsDef}): SSRQuery =>\n    createSSRQuery('${api.modelName}', '${api.methodName}', [${paramNames}], ['${modelName}', '${serviceMethodName}']);\n          `.trim(),\n        );\n      }\n\n      namespaces.push(\n        `\nexport namespace ${modelName}Service {\n${functions.join(\"\\n\\n\")}\n}\n      `.trim(),\n      );\n    }\n\n    return {\n      ...this.getTargetAndPath(),\n      body: namespaces.join(\"\\n\\n\"),\n      importKeys: diff(unique(importKeys), typeParamNames),\n      customHeaders: [\n        \"/** biome-ignore-all lint: generated는 무시 */\",\n        \"/** biome-ignore-all assist: generated는 무시 */\",\n        \"\",\n        `import type { SSRQuery } from 'sonamu/ssr';`,\n        \"\",\n        `// SSRQuery 헬퍼 함수`,\n        `function createSSRQuery(modelName: string, methodName: string, params: any[], serviceKey: [string, string]): SSRQuery {`,\n        `  return { modelName, methodName, params, serviceKey, __brand: 'SSRQuery' } as SSRQuery;`,\n        `}`,\n        \"\",\n      ],\n    };\n  }\n}\n"],"names":["inflection","diff","unique","apiParamToTsCode","apiParamTypeToTsType","Sonamu","ApiParamType","Template","Template__queries","getTargetAndPath","dir","config","api","target","path","render","apis","syncer","queryApis","filter","options","clients","includes","apisByModel","Map","modelName","replace","has","set","get","push","namespaces","importKeys","typeParamNames","modelApis","functions","paramsWithoutContext","parameters","param","isContext","type","isRefKnex","optional","name","startsWith","concat","typeParameters","map","tp","id","typeParametersAsTsType","typeParam","join","typeParamsDef","paramsDef","paramNames","p","serviceMethodName","resourceName","camelize","methodName","trim","body","customHeaders"],"mappings":"AAAA,OAAOA,gBAAgB,aAAa;AACpC,SAASC,IAAI,EAAEC,MAAM,QAAQ,UAAU;AACvC,SAASC,gBAAgB,EAAEC,oBAAoB,QAAQ,+BAA4B;AAEnF,SAASC,MAAM,QAAQ,sBAAmB;AAE1C,SAASC,YAAY,QAAQ,uBAAoB;AACjD,SAASC,QAAQ,QAAQ,iBAAc;AAEvC,OAAO,MAAMC,0BAA0BD;IACrC,aAAc;QACZ,KAAK,CAAC;IACR;IAEAE,mBAAmB;QACjB,MAAM,EAAEC,GAAG,EAAE,GAAGL,OAAOM,MAAM,CAACC,GAAG;QACjC,OAAO;YACLC,QAAQ,GAAGH,IAAI,gBAAgB,CAAC;YAChCI,MAAM,CAAC,oBAAoB,CAAC;QAC9B;IACF;IAEAC,OAAO,EAA8B,EAAE;QACrC,MAAM,EAAEC,IAAI,EAAE,GAAGX,OAAOY,MAAM;QAE9B,+BAA+B;QAC/B,MAAMC,YAAYF,KAAKG,MAAM,CAAC,CAACP,MAAQA,IAAIQ,OAAO,CAACC,OAAO,EAAEC,SAAS;QAErE,WAAW;QACX,MAAMC,cAAc,IAAIC;QACxB,KAAK,MAAMZ,OAAOM,UAAW;YAC3B,MAAMO,YAAYb,IAAIa,SAAS,CAACC,OAAO,CAAC,UAAU,IAAIA,OAAO,CAAC,UAAU;YACxE,IAAI,CAACH,YAAYI,GAAG,CAACF,YAAY;gBAC/BF,YAAYK,GAAG,CAACH,WAAW,EAAE;YAC/B;YACAF,YAAYM,GAAG,CAACJ,YAAYK,KAAKlB;QACnC;QAEA,MAAMmB,aAAuB,EAAE;QAC/B,MAAMC,aAAuB,EAAE;QAC/B,IAAIC,iBAA2B,EAAE;QAEjC,KAAK,MAAM,CAACR,WAAWS,UAAU,IAAIX,YAAa;YAChD,MAAMY,YAAsB,EAAE;YAE9B,KAAK,MAAMvB,OAAOsB,UAAW;gBAC3B,mBAAmB;gBACnB,MAAME,uBAAuBxB,IAAIyB,UAAU,CAAClB,MAAM,CAChD,CAACmB,QACC,CAAChC,aAAaiC,SAAS,CAACD,MAAME,IAAI,KAClC,CAAClC,aAAamC,SAAS,CAACH,MAAME,IAAI,KAClC,CAAEF,CAAAA,MAAMI,QAAQ,KAAK,QAAQJ,MAAMK,IAAI,CAACC,UAAU,CAAC,IAAG;gBAG1D,gBAAgB;gBAChBX,iBAAiBA,eAAeY,MAAM,CAACjC,IAAIkC,cAAc,CAACC,GAAG,CAAC,CAACC,KAAOA,GAAGC,EAAE;gBAE3E,aAAa;gBACb,MAAMC,yBAAyBtC,IAAIkC,cAAc,CAC9CC,GAAG,CAAC,CAACI;oBACJ,OAAO/C,qBAAqB+C,WAAWnB;gBACzC,GACCoB,IAAI,CAAC;gBACR,MAAMC,gBAAgBH,yBAAyB,CAAC,CAAC,EAAEA,uBAAuB,CAAC,CAAC,GAAG;gBAE/E,UAAU;gBACV,MAAMI,YAAYnD,iBAAiBiC,sBAAsBJ;gBACzD,MAAMuB,aAAanB,qBAAqBW,GAAG,CAAC,CAACS,IAAMA,EAAEb,IAAI,EAAES,IAAI,CAAC;gBAEhE,sDAAsD;gBACtD,MAAMK,oBAAoB7C,IAAIQ,OAAO,CAACsC,YAAY,GAC9C,CAAC,GAAG,EAAE1D,WAAW2D,QAAQ,CAAC/C,IAAIQ,OAAO,CAACsC,YAAY,GAAG,GACrD9C,IAAIgD,UAAU;gBAElB,6CAA6C;gBAC7CzB,UAAUL,IAAI,CACZ,CAAC;eACI,EAAE2B,kBAAkB,GAAG,EAAEJ,cAAc,CAAC,EAAEC,UAAU;oBAC/C,EAAE1C,IAAIa,SAAS,CAAC,IAAI,EAAEb,IAAIgD,UAAU,CAAC,IAAI,EAAEL,WAAW,KAAK,EAAE9B,UAAU,IAAI,EAAEgC,kBAAkB;UACzG,CAAC,CAACI,IAAI;YAEV;YAEA9B,WAAWD,IAAI,CACb,CAAC;iBACQ,EAAEL,UAAU;AAC7B,EAAEU,UAAUiB,IAAI,CAAC,QAAQ;;MAEnB,CAAC,CAACS,IAAI;QAER;QAEA,OAAO;YACL,GAAG,IAAI,CAACpD,gBAAgB,EAAE;YAC1BqD,MAAM/B,WAAWqB,IAAI,CAAC;YACtBpB,YAAY/B,KAAKC,OAAO8B,aAAaC;YACrC8B,eAAe;gBACb;gBACA;gBACA;gBACA,CAAC,2CAA2C,CAAC;gBAC7C;gBACA,CAAC,iBAAiB,CAAC;gBACnB,CAAC,uHAAuH,CAAC;gBACzH,CAAC,wFAAwF,CAAC;gBAC1F,CAAC,CAAC,CAAC;gBACH;aACD;QACH;IACF;AACF"}
91
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"sources":["../../../src/template/implementations/queries.template.ts"],"sourcesContent":["import inflection from \"inflection\";\nimport { diff, unique } from \"radashi\";\nimport { apiParamToTsCode, apiParamTypeToTsType } from \"../../api/code-converters\";\nimport type { ExtendedApi } from \"../../api/decorators\";\nimport { Sonamu } from \"../../api/sonamu\";\nimport type { TemplateOptions } from \"../../types/types\";\nimport { ApiParamType } from \"../../types/types\";\nimport { Template } from \"../template\";\n\nexport class Template__queries extends Template {\n  constructor() {\n    super(\"queries\");\n  }\n\n  getTargetAndPath() {\n    const { dir } = Sonamu.config.api;\n    return {\n      target: `${dir}/src/application`,\n      path: `queries.generated.ts`,\n    };\n  }\n\n  render({}: TemplateOptions[\"queries\"]) {\n    const { apis } = Sonamu.syncer;\n\n    // tanstack-query를 포함한 API만 필터링\n    const queryApis = apis.filter((api) => api.options.clients?.includes(\"tanstack-query\"));\n\n    // 모델별로 그룹화\n    const apisByModel = new Map<string, ExtendedApi[]>();\n    for (const api of queryApis) {\n      const modelName = api.modelName.replace(/Model$/, \"\").replace(/Frame$/, \"\");\n      if (!apisByModel.has(modelName)) {\n        apisByModel.set(modelName, []);\n      }\n      apisByModel.get(modelName)?.push(api);\n    }\n\n    const namespaces: string[] = [];\n    const importKeys: string[] = [];\n    let typeParamNames: string[] = [];\n\n    for (const [modelName, modelApis] of apisByModel) {\n      const functions: string[] = [];\n\n      for (const api of modelApis) {\n        // Context 제외한 파라미터\n        const paramsWithoutContext = api.parameters.filter(\n          (param) =>\n            !ApiParamType.isContext(param.type) &&\n            !ApiParamType.isRefKnex(param.type) &&\n            !(param.optional === true && param.name.startsWith(\"_\")),\n        );\n\n        // 타입 파라미터 이름 수집\n        typeParamNames = typeParamNames.concat(api.typeParameters.map((tp) => tp.id));\n\n        // 타입 파라미터 정의\n        const typeParametersAsTsType = api.typeParameters\n          .map((typeParam) => {\n            return apiParamTypeToTsType(typeParam, importKeys);\n          })\n          .join(\", \");\n        const typeParamsDef = typeParametersAsTsType ? `<${typeParametersAsTsType}>` : \"\";\n\n        // 파라미터 정의\n        const paramsDef = apiParamToTsCode(paramsWithoutContext, importKeys);\n        const paramNames = paramsWithoutContext.map((p) => p.name).join(\", \");\n\n        // serviceMethodName 계산 (services.template.ts와 동일한 로직)\n        const serviceMethodName = api.options.resourceName\n          ? `get${inflection.camelize(api.options.resourceName)}`\n          : api.methodName;\n\n        // SSRQuery 함수 생성 (함수명도 serviceMethodName 사용)\n        functions.push(\n          `\n  export const ${serviceMethodName} = ${typeParamsDef}(${paramsDef}): SSRQuery =>\n    createSSRQuery('${api.modelName}', '${api.methodName}', [${paramNames}], ['${modelName}', '${serviceMethodName}']);\n          `.trim(),\n        );\n      }\n\n      namespaces.push(\n        `\nexport namespace ${modelName}Service {\n${functions.join(\"\\n\\n\")}\n}\n      `.trim(),\n      );\n    }\n\n    // tanstack-query API가 없으면 헬퍼 함수와 import를 포함하지 않습니다.\n    // 새 프로젝트에서 첫 빌드 시 sync가 아직 안 되어 namespace가 비어있을 수 있고,\n    // 이때 createSSRQuery가 unused로 빌드 에러가 발생하는 것을 방지합니다.\n    const hasQueries = namespaces.length > 0;\n\n    return {\n      ...this.getTargetAndPath(),\n      body: namespaces.join(\"\\n\\n\"),\n      importKeys: diff(unique(importKeys), typeParamNames),\n      customHeaders: hasQueries\n        ? [\n            \"/** biome-ignore-all lint: generated는 무시 */\",\n            \"/** biome-ignore-all assist: generated는 무시 */\",\n            \"\",\n            `import type { SSRQuery } from 'sonamu/ssr';`,\n            \"\",\n            `// SSRQuery 헬퍼 함수`,\n            `function createSSRQuery(modelName: string, methodName: string, params: any[], serviceKey: [string, string]): SSRQuery {`,\n            `  return { modelName, methodName, params, serviceKey, __brand: 'SSRQuery' } as SSRQuery;`,\n            `}`,\n            \"\",\n          ]\n        : [\n            \"/** biome-ignore-all lint: generated는 무시 */\",\n            \"/** biome-ignore-all assist: generated는 무시 */\",\n            \"\",\n          ],\n    };\n  }\n}\n"],"names":["inflection","diff","unique","apiParamToTsCode","apiParamTypeToTsType","Sonamu","ApiParamType","Template","Template__queries","getTargetAndPath","dir","config","api","target","path","render","apis","syncer","queryApis","filter","options","clients","includes","apisByModel","Map","modelName","replace","has","set","get","push","namespaces","importKeys","typeParamNames","modelApis","functions","paramsWithoutContext","parameters","param","isContext","type","isRefKnex","optional","name","startsWith","concat","typeParameters","map","tp","id","typeParametersAsTsType","typeParam","join","typeParamsDef","paramsDef","paramNames","p","serviceMethodName","resourceName","camelize","methodName","trim","hasQueries","length","body","customHeaders"],"mappings":"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"}
@@ -1 +1 @@
1
- {"version":3,"file":"view_form.template.d.ts","sourceRoot":"","sources":["../../../src/template/implementations/view_form.template.ts"],"names":[],"mappings":"AAGA,OAAO,EAAiB,KAAK,iBAAiB,EAAE,MAAM,6BAA6B,CAAC;AACpF,OAAO,KAAK,EAAE,aAAa,EAAE,WAAW,EAAE,eAAe,EAAE,MAAM,mBAAmB,CAAC;AAGrF,OAAO,EAAE,QAAQ,EAAE,MAAM,aAAa,CAAC;AAEvC,qBAAa,mBAAoB,SAAQ,QAAQ;;IAK/C,gBAAgB,CAAC,KAAK,EAAE,iBAAiB;;;;IAOhC,mBAAmB,IAAI,MAAM,EAAE,GAAG,IAAI;IAI/C,MAAM,CAAC,IAAI,EAAE,MAAM,EAAE,KAAK,CAAC,EAAE,MAAM,GAAG,MAAM;IAO5C,MAAM,CAAC,IAAI,EAAE,MAAM,EAAE,KAAK,CAAC,EAAE,MAAM,GAAG,MAAM;IAI5C,kBAAkB,CAAC,QAAQ,EAAE,MAAM,EAAE,GAAG,EAAE,aAAa;IAmBvD,YAAY,CAAC,QAAQ,EAAE,MAAM,EAAE,GAAG,EAAE,aAAa,EAAE,KAAK,EAAE,iBAAiB,GAAG,MAAM;IAqEpF,mBAAmB,CAAC,OAAO,EAAE,aAAa,EAAE,GAAG,MAAM;IAmC/C,MAAM,CAAC,EAAE,QAAQ,EAAE,EAAE,eAAe,CAAC,WAAW,CAAC;;;;;;;;;;CAoWxD"}
1
+ {"version":3,"file":"view_form.template.d.ts","sourceRoot":"","sources":["../../../src/template/implementations/view_form.template.ts"],"names":[],"mappings":"AAGA,OAAO,EAAiB,KAAK,iBAAiB,EAAE,MAAM,6BAA6B,CAAC;AACpF,OAAO,KAAK,EAAE,aAAa,EAAE,WAAW,EAAE,eAAe,EAAE,MAAM,mBAAmB,CAAC;AAGrF,OAAO,EAAE,QAAQ,EAAE,MAAM,aAAa,CAAC;AAEvC,qBAAa,mBAAoB,SAAQ,QAAQ;;IAK/C,gBAAgB,CAAC,KAAK,EAAE,iBAAiB;;;;IAOhC,mBAAmB,IAAI,MAAM,EAAE,GAAG,IAAI;IAI/C,MAAM,CAAC,IAAI,EAAE,MAAM,EAAE,KAAK,CAAC,EAAE,MAAM,GAAG,MAAM;IAO5C,MAAM,CAAC,IAAI,EAAE,MAAM,EAAE,KAAK,CAAC,EAAE,MAAM,GAAG,MAAM;IAI5C,kBAAkB,CAAC,QAAQ,EAAE,MAAM,EAAE,GAAG,EAAE,aAAa;IAmBvD,YAAY,CAAC,QAAQ,EAAE,MAAM,EAAE,GAAG,EAAE,aAAa,EAAE,KAAK,EAAE,iBAAiB,GAAG,MAAM;IAqEpF,mBAAmB,CAAC,OAAO,EAAE,aAAa,EAAE,GAAG,MAAM;IAqC/C,MAAM,CAAC,EAAE,QAAQ,EAAE,EAAE,eAAe,CAAC,WAAW,CAAC;;;;;;;;;;CAoWxD"}
@@ -130,7 +130,9 @@ export class Template__view_form extends Template {
130
130
  value = Object.keys(col.zodType.enum)[0];
131
131
  } else if (col.zodType instanceof z.ZodBoolean) {
132
132
  value = false;
133
- } else if (col.zodType instanceof z.ZodString) {
133
+ } else if (col.zodType instanceof z.core.$ZodString) {
134
+ // NOTE: z.ZodString으로 비교하면 z.url(), z.email() 등의 타입에서 문제가 생기므로 z.core.$ZodString으로 비교함
135
+ // FIXME: email이나 url 타입 등에 대한 처리가 필요함
134
136
  if (col.renderType === "string-datetime") {
135
137
  value = "now()";
136
138
  } else {
@@ -476,4 +478,4 @@ ${columns.filter((col)=>col.name !== "created_at").map((col)=>{
476
478
  }
477
479
  }
478
480
 
479
- //# sourceMappingURL=data:application/json;base64,{"version":3,"sources":["../../../src/template/implementations/view_form.template.ts"],"sourcesContent":["import inflection from \"inflection\";\nimport { unique } from \"radashi\";\nimport { z } from \"zod\";\nimport { EntityManager, type EntityNamesRecord } from \"../../entity/entity-manager\";\nimport type { RenderingNode, TemplateKey, TemplateOptions } from \"../../types/types\";\nimport { getEnumInfoFromColName, getRelationPropFromColName } from \"../helpers\";\nimport type { RenderedTemplate } from \"../template\";\nimport { Template } from \"../template\";\n\nexport class Template__view_form extends Template {\n  constructor() {\n    super(\"view_form\");\n  }\n\n  getTargetAndPath(names: EntityNamesRecord) {\n    return {\n      target: \"web/src/routes/admin\",\n      path: `${names.fsPlural}/form.tsx`,\n    };\n  }\n\n  override getRequiredDictKeys(): string[] | null {\n    return [\"entity.create\", \"entity.edit\", \"common.backToList\", \"form.createdAt\", \"common.save\"];\n  }\n\n  wrapFC(body: string, label?: string): string {\n    return [\n      `<div className=\"space-y-2\">${label ? `\\n  <Label>${label}</Label>` : \"\"}`,\n      `  ${body}`,\n      `</div>`,\n    ].join(\"\\n\");\n  }\n  wrapFG(body: string, label?: string): string {\n    return this.wrapFC(body, label);\n  }\n\n  renderColumnImport(entityId: string, col: RenderingNode) {\n    if (col.renderType === \"enums\") {\n      const { id, targetEntityNames } = getEnumInfoFromColName(entityId, col.name);\n      const componentId = `${id}Select`;\n      return `import { ${componentId} } from \"@/components/${targetEntityNames.fs}/${componentId}\";`;\n    } else if (col.renderType === \"number-fk_id\") {\n      try {\n        const relProp = getRelationPropFromColName(entityId, col.name.replace(\"_id\", \"\"));\n        const targetNames = EntityManager.getNamesFromId(relProp.with);\n        const componentId = `${relProp.with}IdAsyncSelect`;\n        return `import { ${componentId} } from \"@/components/${targetNames.fs}/${componentId}\";`;\n      } catch {\n        return \"\";\n      }\n    } else {\n      throw new Error(`렌더 불가능한 임포트 ${col.name} ${col.renderType}`);\n    }\n  }\n\n  renderColumn(entityId: string, col: RenderingNode, names: EntityNamesRecord): string {\n    const regExpr = `{...register(\"${col.name}\")}`;\n    const placeholder = `{SD(\"entity.${entityId}.${col.name}\")}`;\n\n    switch (col.renderType) {\n      case \"string-plain\":\n        if (col.zodType instanceof z.ZodString && (col.zodType.maxLength ?? 0) <= 256) {\n          return `<Input className=\"h-8 text-xs bg-white\" placeholder=${placeholder} ${regExpr} />`;\n        } else {\n          return `<Textarea className=\"text-xs bg-white\" rows={4} placeholder=${placeholder} ${regExpr} />`;\n        }\n      case \"datetime\":\n        return `<DateInput\n                    className=\"h-8 text-xs bg-white\"\n                    ${regExpr}\n                  />`;\n      case \"number-id\":\n        return `<input type=\"hidden\" ${regExpr} />`;\n      case \"number-plain\":\n        return `<Input type=\"number\" className=\"h-8 text-xs bg-white\" placeholder=${placeholder} ${regExpr} />`;\n      case \"boolean\":\n        return `<Switch ${regExpr} />`;\n      case \"json-sonamufile\":\n        return `<FileInput\n                    uploadMode=\"lazy\"\n                    viewMode=\"image\"\n                    previewSize=\"md\"\n                    ${regExpr}\n                  />`;\n      case \"json-sonamufile-array\":\n        return `<FileInput\n                    multiple={true}\n                    uploadMode=\"lazy\"\n                    viewMode=\"image\"\n                    previewSize=\"md\"\n                    maxFiles={10}\n                    ${regExpr}\n                  />`;\n      case \"enums\":\n        try {\n          let enumId: string;\n          if (col.name === \"orderBy\") {\n            enumId = `${names.capital}${inflection.camelize(col.name)}Select`;\n          } else {\n            const { id } = getEnumInfoFromColName(entityId, col.name);\n            enumId = `${id}Select`;\n          }\n          return `<${enumId} ${regExpr} ${col.optional || col.nullable ? \"clearable\" : \"\"} />`;\n        } catch {\n          return `<Input className=\"h-8 text-xs bg-white\" ${regExpr} />`;\n        }\n      case \"number-fk_id\":\n        try {\n          const relProp = getRelationPropFromColName(entityId, col.name.replace(\"_id\", \"\"));\n          const fkId = `${relProp.with}IdAsyncSelect`;\n          return `<${fkId} subset=\"A\" ${regExpr} ${\n            col.optional || col.nullable ? \"clearable\" : \"\"\n          } className=\"h-8 text-xs\" />`;\n        } catch {\n          return `<Input type=\"number\" className=\"h-8 text-xs bg-white\" placeholder=${placeholder} ${regExpr} />`;\n        }\n      case \"array\":\n      case \"object\":\n        return `<Input className=\"h-8 text-xs bg-white\" placeholder=\"${col.name}\" ${regExpr} />`;\n      default:\n        return `<Input className=\"h-8 text-xs bg-white\" ${regExpr} />`;\n    }\n  }\n\n  resolveDefaultValue(columns: RenderingNode[]): object {\n    return columns.reduce(\n      (result, col) => {\n        if (col.optional) {\n          return result;\n        }\n\n        let value: unknown;\n        if (col.nullable === true) {\n          value = null;\n        } else if (col.zodType instanceof z.ZodNumber) {\n          value = 0;\n        } else if (col.zodType instanceof z.ZodEnum) {\n          value = Object.keys(col.zodType.enum)[0];\n        } else if (col.zodType instanceof z.ZodBoolean) {\n          value = false;\n        } else if (col.zodType instanceof z.ZodString) {\n          if (col.renderType === \"string-datetime\") {\n            value = \"now()\";\n          } else {\n            value = \"\";\n          }\n        } else if (col.zodType instanceof z.ZodArray) {\n          value = [];\n        } else if (col.zodType instanceof z.ZodObject) {\n          value = {};\n        }\n\n        result[col.name] = value;\n        return result;\n      },\n      {} as { [key: string]: unknown },\n    );\n  }\n\n  async render({ entityId }: TemplateOptions[\"view_form\"]) {\n    const entity = EntityManager.get(entityId);\n    const names = EntityManager.getNamesFromId(entityId);\n\n    // SaveParams 타입을 로드하여 saveParamsNode 생성\n    const { loadTypes } = await import(\"../../syncer/module-loader\");\n    const loadedTypes = await loadTypes();\n    const SaveParamsZodType = loadedTypes[`${entityId}SaveParams`];\n\n    if (!SaveParamsZodType) {\n      throw new Error(`SaveParams for ${entityId} not found. Did you run 'sonamu sync'?`);\n    }\n\n    // Zod 타입을 RenderingNode로 변환\n    const { zodTypeToRenderingNode } = await import(\"../zod-converter\");\n    const saveParamsNode = zodTypeToRenderingNode(SaveParamsZodType);\n\n    const columns = ((saveParamsNode?.children ?? []) as RenderingNode[])\n      .filter((col) => col.name !== \"id\")\n      .map((col) => {\n        const propCandidate = entity.props.find((prop) => prop.name === col.name);\n        col.label = propCandidate?.desc ?? col.label;\n        return col;\n      });\n\n    const defaultValue = this.resolveDefaultValue(columns);\n\n    // 프리 템플릿\n    const preTemplates: RenderedTemplate[\"preTemplates\"] = (columns as RenderingNode[])\n      .filter((col) => {\n        if (col.name === \"id\") {\n          return false;\n        } else if (col.name.endsWith(\"_id\") || col.renderType === \"number-id\") {\n          try {\n            getRelationPropFromColName(entityId, col.name.replace(\"_id\", \"\"));\n            return true;\n          } catch {\n            return false;\n          }\n        } else if (col.renderType === \"enums\") {\n          try {\n            getEnumInfoFromColName(entityId, col.name);\n            return true;\n          } catch {\n            return false;\n          }\n        }\n        return false;\n      })\n      .map((col) => {\n        let key: TemplateKey;\n        let targetMdId = entityId;\n        let enumId: string | undefined;\n        if (col.renderType === \"enums\") {\n          key = \"view_enums_select\";\n          const { targetEntityNames: targetMDNames, id } = getEnumInfoFromColName(\n            entityId,\n            col.name,\n          );\n          targetMdId = targetMDNames.capital;\n          enumId = id;\n        } else {\n          key = \"view_id_async_select\";\n          const relProp = getRelationPropFromColName(entityId, col.name.replace(\"_id\", \"\"));\n          targetMdId = relProp.with;\n        }\n\n        return {\n          key: key as TemplateKey,\n          options: {\n            entityId: targetMdId,\n            node: col,\n            enumId,\n          },\n        };\n      })\n      .filter((preTemplate) => {\n        if (preTemplate.key === \"view_id_async_select\") {\n          try {\n            EntityManager.get(preTemplate.options.entityId);\n            return true;\n          } catch {\n            return false;\n          }\n        }\n        return true;\n      });\n\n    return {\n      ...this.getTargetAndPath(names),\n      body: `\nimport {\n  Button,\n  Card,\n  CardContent,\n  CardHeader,\n  CardTitle,\n  Input,${columns.some((col) => col.renderType === \"string-plain\" && col.zodType instanceof z.ZodString && (col.zodType.maxLength ?? 0) > 256) ? \"\\n  Textarea,\" : \"\"}${columns.some((col) => col.renderType === \"enums\") ? \"\\n  Select,\\n  SelectContent,\\n  SelectItem,\\n  SelectTrigger,\\n  SelectValue,\" : \"\"}${columns.some((col) => col.renderType === \"boolean\") ? \"\\n  Switch,\" : \"\"}${columns.some((col) => [\"json-sonamufile\", \"json-sonamufile-array\"].includes(col.renderType)) ? \"\\n  FileInput,\" : \"\"}${columns.some((col) => [\"string-datetime\", \"string-date\", \"datetime\"].includes(col.renderType)) ? \"\\n  DateInput,\" : \"\"}\n} from \"@sonamu-kit/react-components/components\";\nimport { useTypeForm } from \"@sonamu-kit/react-components/lib\";\nimport { useQueryClient } from \"@tanstack/react-query\";\nimport { createFileRoute, useRouter } from \"@tanstack/react-router\";\nimport { useEffect } from \"react\";\nimport { z } from \"zod\";\nimport { ${names.capital}Service } from \"@/services/services.generated\";\nimport type { ${names.capital}SubsetA } from \"@/services/sonamu.generated\";${\n        columns.filter((col) => col.renderType === \"enums\").length > 0\n          ? \"\\nimport { \" +\n            unique(\n              columns\n                .filter((col) => col.renderType === \"enums\")\n                .map((col) => {\n                  try {\n                    const { id } = getEnumInfoFromColName(entityId, col.name);\n                    return `${id}, ${id}Label`;\n                  } catch {\n                    return \"\";\n                  }\n                }),\n            )\n              .filter(Boolean)\n              .join(\", \") +\n            ' } from \"@/services/sonamu.generated\";'\n          : \"\"\n      }\nimport { defaultCatch } from \"@/services/sonamu.shared\";\nimport { ${names.capital}SaveParams } from \"@/services/${names.fs}/${names.fs}.types\";\n${unique(\n  columns\n    .filter((col) => [\"number-fk_id\", \"enums\"].includes(col.renderType))\n    .map((col) => {\n      return this.renderColumnImport(entityId, col);\n    }),\n).join(\"\\n\")}\nimport { SD } from \"@/i18n/sd.generated\";\n\nimport ArrowLeftIcon from \"~icons/lucide/arrow-left\";\nimport SaveIcon from \"~icons/lucide/save\";\nimport FormIcon from \"~icons/mdi/form-select\";\n\nconst formSearchSchema = z.object({\n  id: z.number().optional(),\n});\n\nexport const Route = createFileRoute(\"/admin/${names.fsPlural}/form\")({\n  validateSearch: formSearchSchema,\n  component: ${names.capitalPlural}FormPage,\n});\n\nfunction ${names.capitalPlural}FormPage() {\n  const { id } = Route.useSearch();\n  return <${names.capitalPlural}Form id={id} />;\n}\n\ntype ${names.capitalPlural}FormProps = {\n  id?: number;\n  mode?: \"page\" | \"modal\";\n};\n\nexport function ${names.capitalPlural}Form({ id, mode }: ${names.capitalPlural}FormProps) {\n  const router = useRouter();\n  const queryClient = useQueryClient();\n\n  const { form, setForm, register${columns.some((col) => [\"json-sonamufile\", \"json-sonamufile-array\"].includes(col.renderType)) ? \", submit\" : \"\"} } = useTypeForm(${names.capital}SaveParams, ${JSON.stringify(defaultValue).replace(/\"now\\(\\)\"/g, '\"\"')});\n${(() => {\n  const hasDatetime = columns.some((col) => col.renderType === \"string-datetime\");\n  const hasDate = columns.some((col) => col.renderType === \"string-date\");\n  if (!hasDatetime && !hasDate) return \"\";\n\n  let helpers = \"\\n\";\n  if (hasDatetime) {\n    helpers += `  // datetime-local 형식으로 변환 (YYYY-MM-DDTHH:MM)\n  const toDatetimeLocalString = (date: Date | string | null | undefined): string => {\n    if (!date) return \"\";\n    const d = typeof date === \"string\" ? new Date(date) : date;\n    return d.toISOString().slice(0, 16);\n  };\n\n  // datetime-local 문자열을 Date로 변환\n  const fromDatetimeLocalString = (value: string): Date | null => {\n    if (!value) return null;\n    return new Date(value);\n  };\n`;\n  }\n  if (hasDate) {\n    helpers += `  // date 형식으로 변환 (YYYY-MM-DD)\n  const toDateString = (date: Date | string | null | undefined): string => {\n    if (!date) return \"\";\n    const d = typeof date === \"string\" ? new Date(date) : date;\n    return d.toISOString().split(\"T\")[0];\n  };\n\n  // date 문자열을 Date로 변환\n  const fromDateString = (value: string): Date | null => {\n    if (!value) return null;\n    return new Date(value);\n  };\n`;\n  }\n  return helpers;\n})()}\n  useEffect(() => {\n    if (id) {\n      ${names.capital}Service.get${names.capital}(\"A\", id).then((row) => {\n        setForm((prevForm) => ({\n          ...prevForm,\n          ...row,${(() => {\n            // relation 필드들을 찾아서 변환 코드 생성\n            const relationFields = columns\n              .filter((col) => col.renderType === \"number-fk_id\")\n              .map((col) => {\n                const relationName = col.name.replace(/_id$/, \"\");\n                if (col.nullable) {\n                  return `\\n          ${col.name}: row.${relationName}?.id ?? null,`;\n                } else {\n                  return `\\n          ${col.name}: row.${relationName}?.id,`;\n                }\n              })\n              .join(\"\");\n            return relationFields;\n          })()}\n        }));\n      });\n    }\n  }, [id, setForm]);\n\n  const saveMutation = ${names.capital}Service.useSaveMutation();\n  const handleSubmit = ${\n    columns.some((col) => [\"json-sonamufile\", \"json-sonamufile-array\"].includes(col.renderType))\n      ? `submit(async (form) => {\n    saveMutation.mutate(\n      { spa: [form] },\n      {\n        onSuccess: () => {\n          queryClient.invalidateQueries({\n            queryKey: [\"${names.capital}\"],\n          });\n\n          if (mode === \"modal\") {\n            // modal mode\n          } else {\n            router.navigate({ to: \"/admin/${names.fsPlural}\" });\n          }\n        },\n        onError: defaultCatch,\n      },\n    );\n  })`\n      : `() => {\n    saveMutation.mutate(\n      { spa: [form] },\n      {\n        onSuccess: () => {\n          queryClient.invalidateQueries({\n            queryKey: [\"${names.capital}\"],\n          });\n\n          if (mode === \"modal\") {\n            // modal mode\n          } else {\n            router.navigate({ to: \"/admin/${names.fsPlural}\" });\n          }\n        },\n        onError: defaultCatch,\n      },\n    );\n  }`\n  };\n\n  const PAGE = {\n    title: id ? SD(\"entity.edit\")(SD(\"entity.${entityId}\"), id) : SD(\"entity.create\")(SD(\"entity.${entityId}\")),\n  };\n\n  return (\n    <div className=\"flex-1 overflow-auto\">\n      <div className=\"max-w-[1800px] mx-auto p-8\">\n        <div className=\"space-y-6 mb-8\">\n          {/* Header */}\n          <div className=\"flex items-center justify-between\">\n            <div className=\"flex items-center gap-2\">\n              <FormIcon className=\"h-5 w-5\" />\n              <span className=\"text-lg font-semibold h-5\">{PAGE.title}</span>\n            </div>\n            {mode !== \"modal\" && (\n              <Button\n                variant=\"outline\"\n                onClick={() => router.navigate({ to: \"/admin/${names.fsPlural}\" })}\n                icon={<ArrowLeftIcon />}\n              >\n                {SD(\"common.backToList\")}\n              </Button>\n            )}\n          </div>\n\n          {/* Form Card */}\n          <Card className=\"border-border/40 bg-gray-50 shadow-sm\">\n            <CardHeader className=\"px-4 border-b border-gray-200 flex items-center\">\n              <CardTitle className=\"text-sm font-medium leading-none m-0\">\n                {PAGE.title}\n              </CardTitle>\n            </CardHeader>\n            <CardContent className=\"p-6\">\n              <div className=\"space-y-6\">\n${columns\n  .filter((col) => col.name !== \"created_at\")\n  .map((col) => {\n    const label = (() => {\n      if (col.label.endsWith(\"Id\")) {\n        try {\n          const entity = EntityManager.get(col.label.replace(\"Id\", \"\"));\n          return entity.title ?? col.label;\n        } catch {\n          return col.label;\n        }\n      }\n      return col.label;\n    })();\n    return `                {/* ${label} */}\n                <div className=\"space-y-2\">\n                  <label className=\"block text-xs mb-1 text-gray-600\">{SD(\"entity.${entityId}.${col.name}\")}</label>\n                  ${this.renderColumn(entityId, col, names)}\n                </div>`;\n  })\n  .join(\"\\n\\n\")}\n\n                {/* Save Button */}\n                <div className=\"flex items-center justify-between pt-4\">\n                  {form.id && form.created_at && (\n                    <div className=\"flex items-center\">\n                      <label className=\"mr-2 text-xs text-gray-600\">{SD(\"form.createdAt\")}:</label>\n                      <span className=\"text-xs text-gray-600\">\n                        {String(form.created_at)}\n                      </span>\n                    </div>\n                  )}\n                  <Button\n                    onClick={handleSubmit}\n                    icon={<SaveIcon />}\n                  >\n                    {SD(\"common.save\")}\n                  </Button>\n                </div>\n              </div>\n            </CardContent>\n          </Card>\n        </div>\n      </div>\n    </div>\n  );\n}\n      `.trim(),\n      importKeys: [],\n      preTemplates,\n    };\n  }\n}\n"],"names":["inflection","unique","z","EntityManager","getEnumInfoFromColName","getRelationPropFromColName","Template","Template__view_form","getTargetAndPath","names","target","path","fsPlural","getRequiredDictKeys","wrapFC","body","label","join","wrapFG","renderColumnImport","entityId","col","renderType","id","targetEntityNames","name","componentId","fs","relProp","replace","targetNames","getNamesFromId","with","Error","renderColumn","regExpr","placeholder","zodType","ZodString","maxLength","enumId","capital","camelize","optional","nullable","fkId","resolveDefaultValue","columns","reduce","result","value","ZodNumber","ZodEnum","Object","keys","enum","ZodBoolean","ZodArray","ZodObject","render","entity","get","loadTypes","loadedTypes","SaveParamsZodType","zodTypeToRenderingNode","saveParamsNode","children","filter","map","propCandidate","props","find","prop","desc","defaultValue","preTemplates","endsWith","key","targetMdId","targetMDNames","options","node","preTemplate","some","includes","length","Boolean","capitalPlural","JSON","stringify","hasDatetime","hasDate","helpers","relationFields","relationName","title","trim","importKeys"],"mappings":"AAAA,OAAOA,gBAAgB,aAAa;AACpC,SAASC,MAAM,QAAQ,UAAU;AACjC,SAASC,CAAC,QAAQ,MAAM;AACxB,SAASC,aAAa,QAAgC,iCAA8B;AAEpF,SAASC,sBAAsB,EAAEC,0BAA0B,QAAQ,gBAAa;AAEhF,SAASC,QAAQ,QAAQ,iBAAc;AAEvC,OAAO,MAAMC,4BAA4BD;IACvC,aAAc;QACZ,KAAK,CAAC;IACR;IAEAE,iBAAiBC,KAAwB,EAAE;QACzC,OAAO;YACLC,QAAQ;YACRC,MAAM,GAAGF,MAAMG,QAAQ,CAAC,SAAS,CAAC;QACpC;IACF;IAESC,sBAAuC;QAC9C,OAAO;YAAC;YAAiB;YAAe;YAAqB;YAAkB;SAAc;IAC/F;IAEAC,OAAOC,IAAY,EAAEC,KAAc,EAAU;QAC3C,OAAO;YACL,CAAC,2BAA2B,EAAEA,QAAQ,CAAC,WAAW,EAAEA,MAAM,QAAQ,CAAC,GAAG,IAAI;YAC1E,CAAC,EAAE,EAAED,MAAM;YACX,CAAC,MAAM,CAAC;SACT,CAACE,IAAI,CAAC;IACT;IACAC,OAAOH,IAAY,EAAEC,KAAc,EAAU;QAC3C,OAAO,IAAI,CAACF,MAAM,CAACC,MAAMC;IAC3B;IAEAG,mBAAmBC,QAAgB,EAAEC,GAAkB,EAAE;QACvD,IAAIA,IAAIC,UAAU,KAAK,SAAS;YAC9B,MAAM,EAAEC,EAAE,EAAEC,iBAAiB,EAAE,GAAGpB,uBAAuBgB,UAAUC,IAAII,IAAI;YAC3E,MAAMC,cAAc,GAAGH,GAAG,MAAM,CAAC;YACjC,OAAO,CAAC,SAAS,EAAEG,YAAY,sBAAsB,EAAEF,kBAAkBG,EAAE,CAAC,CAAC,EAAED,YAAY,EAAE,CAAC;QAChG,OAAO,IAAIL,IAAIC,UAAU,KAAK,gBAAgB;YAC5C,IAAI;gBACF,MAAMM,UAAUvB,2BAA2Be,UAAUC,IAAII,IAAI,CAACI,OAAO,CAAC,OAAO;gBAC7E,MAAMC,cAAc3B,cAAc4B,cAAc,CAACH,QAAQI,IAAI;gBAC7D,MAAMN,cAAc,GAAGE,QAAQI,IAAI,CAAC,aAAa,CAAC;gBAClD,OAAO,CAAC,SAAS,EAAEN,YAAY,sBAAsB,EAAEI,YAAYH,EAAE,CAAC,CAAC,EAAED,YAAY,EAAE,CAAC;YAC1F,EAAE,OAAM;gBACN,OAAO;YACT;QACF,OAAO;YACL,MAAM,IAAIO,MAAM,CAAC,YAAY,EAAEZ,IAAII,IAAI,CAAC,CAAC,EAAEJ,IAAIC,UAAU,EAAE;QAC7D;IACF;IAEAY,aAAad,QAAgB,EAAEC,GAAkB,EAAEZ,KAAwB,EAAU;QACnF,MAAM0B,UAAU,CAAC,cAAc,EAAEd,IAAII,IAAI,CAAC,GAAG,CAAC;QAC9C,MAAMW,cAAc,CAAC,YAAY,EAAEhB,SAAS,CAAC,EAAEC,IAAII,IAAI,CAAC,GAAG,CAAC;QAE5D,OAAQJ,IAAIC,UAAU;YACpB,KAAK;gBACH,IAAID,IAAIgB,OAAO,YAAYnC,EAAEoC,SAAS,IAAI,AAACjB,CAAAA,IAAIgB,OAAO,CAACE,SAAS,IAAI,CAAA,KAAM,KAAK;oBAC7E,OAAO,CAAC,oDAAoD,EAAEH,YAAY,CAAC,EAAED,QAAQ,GAAG,CAAC;gBAC3F,OAAO;oBACL,OAAO,CAAC,4DAA4D,EAAEC,YAAY,CAAC,EAAED,QAAQ,GAAG,CAAC;gBACnG;YACF,KAAK;gBACH,OAAO,CAAC;;oBAEI,EAAEA,QAAQ;oBACV,CAAC;YACf,KAAK;gBACH,OAAO,CAAC,qBAAqB,EAAEA,QAAQ,GAAG,CAAC;YAC7C,KAAK;gBACH,OAAO,CAAC,kEAAkE,EAAEC,YAAY,CAAC,EAAED,QAAQ,GAAG,CAAC;YACzG,KAAK;gBACH,OAAO,CAAC,QAAQ,EAAEA,QAAQ,GAAG,CAAC;YAChC,KAAK;gBACH,OAAO,CAAC;;;;oBAII,EAAEA,QAAQ;oBACV,CAAC;YACf,KAAK;gBACH,OAAO,CAAC;;;;;;oBAMI,EAAEA,QAAQ;oBACV,CAAC;YACf,KAAK;gBACH,IAAI;oBACF,IAAIK;oBACJ,IAAInB,IAAII,IAAI,KAAK,WAAW;wBAC1Be,SAAS,GAAG/B,MAAMgC,OAAO,GAAGzC,WAAW0C,QAAQ,CAACrB,IAAII,IAAI,EAAE,MAAM,CAAC;oBACnE,OAAO;wBACL,MAAM,EAAEF,EAAE,EAAE,GAAGnB,uBAAuBgB,UAAUC,IAAII,IAAI;wBACxDe,SAAS,GAAGjB,GAAG,MAAM,CAAC;oBACxB;oBACA,OAAO,CAAC,CAAC,EAAEiB,OAAO,CAAC,EAAEL,QAAQ,CAAC,EAAEd,IAAIsB,QAAQ,IAAItB,IAAIuB,QAAQ,GAAG,cAAc,GAAG,GAAG,CAAC;gBACtF,EAAE,OAAM;oBACN,OAAO,CAAC,wCAAwC,EAAET,QAAQ,GAAG,CAAC;gBAChE;YACF,KAAK;gBACH,IAAI;oBACF,MAAMP,UAAUvB,2BAA2Be,UAAUC,IAAII,IAAI,CAACI,OAAO,CAAC,OAAO;oBAC7E,MAAMgB,OAAO,GAAGjB,QAAQI,IAAI,CAAC,aAAa,CAAC;oBAC3C,OAAO,CAAC,CAAC,EAAEa,KAAK,YAAY,EAAEV,QAAQ,CAAC,EACrCd,IAAIsB,QAAQ,IAAItB,IAAIuB,QAAQ,GAAG,cAAc,GAC9C,2BAA2B,CAAC;gBAC/B,EAAE,OAAM;oBACN,OAAO,CAAC,kEAAkE,EAAER,YAAY,CAAC,EAAED,QAAQ,GAAG,CAAC;gBACzG;YACF,KAAK;YACL,KAAK;gBACH,OAAO,CAAC,qDAAqD,EAAEd,IAAII,IAAI,CAAC,EAAE,EAAEU,QAAQ,GAAG,CAAC;YAC1F;gBACE,OAAO,CAAC,wCAAwC,EAAEA,QAAQ,GAAG,CAAC;QAClE;IACF;IAEAW,oBAAoBC,OAAwB,EAAU;QACpD,OAAOA,QAAQC,MAAM,CACnB,CAACC,QAAQ5B;YACP,IAAIA,IAAIsB,QAAQ,EAAE;gBAChB,OAAOM;YACT;YAEA,IAAIC;YACJ,IAAI7B,IAAIuB,QAAQ,KAAK,MAAM;gBACzBM,QAAQ;YACV,OAAO,IAAI7B,IAAIgB,OAAO,YAAYnC,EAAEiD,SAAS,EAAE;gBAC7CD,QAAQ;YACV,OAAO,IAAI7B,IAAIgB,OAAO,YAAYnC,EAAEkD,OAAO,EAAE;gBAC3CF,QAAQG,OAAOC,IAAI,CAACjC,IAAIgB,OAAO,CAACkB,IAAI,CAAC,CAAC,EAAE;YAC1C,OAAO,IAAIlC,IAAIgB,OAAO,YAAYnC,EAAEsD,UAAU,EAAE;gBAC9CN,QAAQ;YACV,OAAO,IAAI7B,IAAIgB,OAAO,YAAYnC,EAAEoC,SAAS,EAAE;gBAC7C,IAAIjB,IAAIC,UAAU,KAAK,mBAAmB;oBACxC4B,QAAQ;gBACV,OAAO;oBACLA,QAAQ;gBACV;YACF,OAAO,IAAI7B,IAAIgB,OAAO,YAAYnC,EAAEuD,QAAQ,EAAE;gBAC5CP,QAAQ,EAAE;YACZ,OAAO,IAAI7B,IAAIgB,OAAO,YAAYnC,EAAEwD,SAAS,EAAE;gBAC7CR,QAAQ,CAAC;YACX;YAEAD,MAAM,CAAC5B,IAAII,IAAI,CAAC,GAAGyB;YACnB,OAAOD;QACT,GACA,CAAC;IAEL;IAEA,MAAMU,OAAO,EAAEvC,QAAQ,EAAgC,EAAE;QACvD,MAAMwC,SAASzD,cAAc0D,GAAG,CAACzC;QACjC,MAAMX,QAAQN,cAAc4B,cAAc,CAACX;QAE3C,wCAAwC;QACxC,MAAM,EAAE0C,SAAS,EAAE,GAAG,MAAM,MAAM,CAAC;QACnC,MAAMC,cAAc,MAAMD;QAC1B,MAAME,oBAAoBD,WAAW,CAAC,GAAG3C,SAAS,UAAU,CAAC,CAAC;QAE9D,IAAI,CAAC4C,mBAAmB;YACtB,MAAM,IAAI/B,MAAM,CAAC,eAAe,EAAEb,SAAS,sCAAsC,CAAC;QACpF;QAEA,4BAA4B;QAC5B,MAAM,EAAE6C,sBAAsB,EAAE,GAAG,MAAM,MAAM,CAAC;QAChD,MAAMC,iBAAiBD,uBAAuBD;QAE9C,MAAMjB,UAAU,AAAEmB,CAAAA,gBAAgBC,YAAY,EAAE,AAAD,EAC5CC,MAAM,CAAC,CAAC/C,MAAQA,IAAII,IAAI,KAAK,MAC7B4C,GAAG,CAAC,CAAChD;YACJ,MAAMiD,gBAAgBV,OAAOW,KAAK,CAACC,IAAI,CAAC,CAACC,OAASA,KAAKhD,IAAI,KAAKJ,IAAII,IAAI;YACxEJ,IAAIL,KAAK,GAAGsD,eAAeI,QAAQrD,IAAIL,KAAK;YAC5C,OAAOK;QACT;QAEF,MAAMsD,eAAe,IAAI,CAAC7B,mBAAmB,CAACC;QAE9C,SAAS;QACT,MAAM6B,eAAiD,AAAC7B,QACrDqB,MAAM,CAAC,CAAC/C;YACP,IAAIA,IAAII,IAAI,KAAK,MAAM;gBACrB,OAAO;YACT,OAAO,IAAIJ,IAAII,IAAI,CAACoD,QAAQ,CAAC,UAAUxD,IAAIC,UAAU,KAAK,aAAa;gBACrE,IAAI;oBACFjB,2BAA2Be,UAAUC,IAAII,IAAI,CAACI,OAAO,CAAC,OAAO;oBAC7D,OAAO;gBACT,EAAE,OAAM;oBACN,OAAO;gBACT;YACF,OAAO,IAAIR,IAAIC,UAAU,KAAK,SAAS;gBACrC,IAAI;oBACFlB,uBAAuBgB,UAAUC,IAAII,IAAI;oBACzC,OAAO;gBACT,EAAE,OAAM;oBACN,OAAO;gBACT;YACF;YACA,OAAO;QACT,GACC4C,GAAG,CAAC,CAAChD;YACJ,IAAIyD;YACJ,IAAIC,aAAa3D;YACjB,IAAIoB;YACJ,IAAInB,IAAIC,UAAU,KAAK,SAAS;gBAC9BwD,MAAM;gBACN,MAAM,EAAEtD,mBAAmBwD,aAAa,EAAEzD,EAAE,EAAE,GAAGnB,uBAC/CgB,UACAC,IAAII,IAAI;gBAEVsD,aAAaC,cAAcvC,OAAO;gBAClCD,SAASjB;YACX,OAAO;gBACLuD,MAAM;gBACN,MAAMlD,UAAUvB,2BAA2Be,UAAUC,IAAII,IAAI,CAACI,OAAO,CAAC,OAAO;gBAC7EkD,aAAanD,QAAQI,IAAI;YAC3B;YAEA,OAAO;gBACL8C,KAAKA;gBACLG,SAAS;oBACP7D,UAAU2D;oBACVG,MAAM7D;oBACNmB;gBACF;YACF;QACF,GACC4B,MAAM,CAAC,CAACe;YACP,IAAIA,YAAYL,GAAG,KAAK,wBAAwB;gBAC9C,IAAI;oBACF3E,cAAc0D,GAAG,CAACsB,YAAYF,OAAO,CAAC7D,QAAQ;oBAC9C,OAAO;gBACT,EAAE,OAAM;oBACN,OAAO;gBACT;YACF;YACA,OAAO;QACT;QAEF,OAAO;YACL,GAAG,IAAI,CAACZ,gBAAgB,CAACC,MAAM;YAC/BM,MAAM,CAAC;;;;;;;QAOL,EAAEgC,QAAQqC,IAAI,CAAC,CAAC/D,MAAQA,IAAIC,UAAU,KAAK,kBAAkBD,IAAIgB,OAAO,YAAYnC,EAAEoC,SAAS,IAAI,AAACjB,CAAAA,IAAIgB,OAAO,CAACE,SAAS,IAAI,CAAA,IAAK,OAAO,kBAAkB,KAAKQ,QAAQqC,IAAI,CAAC,CAAC/D,MAAQA,IAAIC,UAAU,KAAK,WAAW,mFAAmF,KAAKyB,QAAQqC,IAAI,CAAC,CAAC/D,MAAQA,IAAIC,UAAU,KAAK,aAAa,gBAAgB,KAAKyB,QAAQqC,IAAI,CAAC,CAAC/D,MAAQ;oBAAC;oBAAmB;iBAAwB,CAACgE,QAAQ,CAAChE,IAAIC,UAAU,KAAK,mBAAmB,KAAKyB,QAAQqC,IAAI,CAAC,CAAC/D,MAAQ;oBAAC;oBAAmB;oBAAe;iBAAW,CAACgE,QAAQ,CAAChE,IAAIC,UAAU,KAAK,mBAAmB,GAAG;;;;;;;SAOpmB,EAAEb,MAAMgC,OAAO,CAAC;cACX,EAAEhC,MAAMgC,OAAO,CAAC,6CAA6C,EACnEM,QAAQqB,MAAM,CAAC,CAAC/C,MAAQA,IAAIC,UAAU,KAAK,SAASgE,MAAM,GAAG,IACzD,gBACArF,OACE8C,QACGqB,MAAM,CAAC,CAAC/C,MAAQA,IAAIC,UAAU,KAAK,SACnC+C,GAAG,CAAC,CAAChD;gBACJ,IAAI;oBACF,MAAM,EAAEE,EAAE,EAAE,GAAGnB,uBAAuBgB,UAAUC,IAAII,IAAI;oBACxD,OAAO,GAAGF,GAAG,EAAE,EAAEA,GAAG,KAAK,CAAC;gBAC5B,EAAE,OAAM;oBACN,OAAO;gBACT;YACF,IAED6C,MAAM,CAACmB,SACPtE,IAAI,CAAC,QACR,2CACA,GACL;;SAEE,EAAER,MAAMgC,OAAO,CAAC,8BAA8B,EAAEhC,MAAMkB,EAAE,CAAC,CAAC,EAAElB,MAAMkB,EAAE,CAAC;AAC9E,EAAE1B,OACA8C,QACGqB,MAAM,CAAC,CAAC/C,MAAQ;oBAAC;oBAAgB;iBAAQ,CAACgE,QAAQ,CAAChE,IAAIC,UAAU,GACjE+C,GAAG,CAAC,CAAChD;gBACJ,OAAO,IAAI,CAACF,kBAAkB,CAACC,UAAUC;YAC3C,IACFJ,IAAI,CAAC,MAAM;;;;;;;;;;;6CAWgC,EAAER,MAAMG,QAAQ,CAAC;;aAEjD,EAAEH,MAAM+E,aAAa,CAAC;;;SAG1B,EAAE/E,MAAM+E,aAAa,CAAC;;UAErB,EAAE/E,MAAM+E,aAAa,CAAC;;;KAG3B,EAAE/E,MAAM+E,aAAa,CAAC;;;;;gBAKX,EAAE/E,MAAM+E,aAAa,CAAC,mBAAmB,EAAE/E,MAAM+E,aAAa,CAAC;;;;iCAI9C,EAAEzC,QAAQqC,IAAI,CAAC,CAAC/D,MAAQ;oBAAC;oBAAmB;iBAAwB,CAACgE,QAAQ,CAAChE,IAAIC,UAAU,KAAK,aAAa,GAAG,iBAAiB,EAAEb,MAAMgC,OAAO,CAAC,YAAY,EAAEgD,KAAKC,SAAS,CAACf,cAAc9C,OAAO,CAAC,cAAc,MAAM;AAC1P,EAAE,AAAC,CAAA;gBACD,MAAM8D,cAAc5C,QAAQqC,IAAI,CAAC,CAAC/D,MAAQA,IAAIC,UAAU,KAAK;gBAC7D,MAAMsE,UAAU7C,QAAQqC,IAAI,CAAC,CAAC/D,MAAQA,IAAIC,UAAU,KAAK;gBACzD,IAAI,CAACqE,eAAe,CAACC,SAAS,OAAO;gBAErC,IAAIC,UAAU;gBACd,IAAIF,aAAa;oBACfE,WAAW,CAAC;;;;;;;;;;;;AAYhB,CAAC;gBACC;gBACA,IAAID,SAAS;oBACXC,WAAW,CAAC;;;;;;;;;;;;AAYhB,CAAC;gBACC;gBACA,OAAOA;YACT,CAAA,IAAK;;;MAGC,EAAEpF,MAAMgC,OAAO,CAAC,WAAW,EAAEhC,MAAMgC,OAAO,CAAC;;;iBAGhC,EAAE,AAAC,CAAA;gBACR,6BAA6B;gBAC7B,MAAMqD,iBAAiB/C,QACpBqB,MAAM,CAAC,CAAC/C,MAAQA,IAAIC,UAAU,KAAK,gBACnC+C,GAAG,CAAC,CAAChD;oBACJ,MAAM0E,eAAe1E,IAAII,IAAI,CAACI,OAAO,CAAC,QAAQ;oBAC9C,IAAIR,IAAIuB,QAAQ,EAAE;wBAChB,OAAO,CAAC,YAAY,EAAEvB,IAAII,IAAI,CAAC,MAAM,EAAEsE,aAAa,aAAa,CAAC;oBACpE,OAAO;wBACL,OAAO,CAAC,YAAY,EAAE1E,IAAII,IAAI,CAAC,MAAM,EAAEsE,aAAa,KAAK,CAAC;oBAC5D;gBACF,GACC9E,IAAI,CAAC;gBACR,OAAO6E;YACT,CAAA,IAAK;;;;;;uBAMQ,EAAErF,MAAMgC,OAAO,CAAC;uBAChB,EACnBM,QAAQqC,IAAI,CAAC,CAAC/D,MAAQ;oBAAC;oBAAmB;iBAAwB,CAACgE,QAAQ,CAAChE,IAAIC,UAAU,KACtF,CAAC;;;;;;wBAMe,EAAEb,MAAMgC,OAAO,CAAC;;;;;;0CAME,EAAEhC,MAAMG,QAAQ,CAAC;;;;;;IAMvD,CAAC,GACG,CAAC;;;;;;wBAMe,EAAEH,MAAMgC,OAAO,CAAC;;;;;;0CAME,EAAEhC,MAAMG,QAAQ,CAAC;;;;;;GAMxD,CAAC,CACD;;;6CAG0C,EAAEQ,SAAS,yCAAyC,EAAEA,SAAS;;;;;;;;;;;;;;;;6DAgB/C,EAAEX,MAAMG,QAAQ,CAAC;;;;;;;;;;;;;;;;;AAiB9E,EAAEmC,QACCqB,MAAM,CAAC,CAAC/C,MAAQA,IAAII,IAAI,KAAK,cAC7B4C,GAAG,CAAC,CAAChD;gBACJ,MAAML,QAAQ,AAAC,CAAA;oBACb,IAAIK,IAAIL,KAAK,CAAC6D,QAAQ,CAAC,OAAO;wBAC5B,IAAI;4BACF,MAAMjB,SAASzD,cAAc0D,GAAG,CAACxC,IAAIL,KAAK,CAACa,OAAO,CAAC,MAAM;4BACzD,OAAO+B,OAAOoC,KAAK,IAAI3E,IAAIL,KAAK;wBAClC,EAAE,OAAM;4BACN,OAAOK,IAAIL,KAAK;wBAClB;oBACF;oBACA,OAAOK,IAAIL,KAAK;gBAClB,CAAA;gBACA,OAAO,CAAC,oBAAoB,EAAEA,MAAM;;kFAE0C,EAAEI,SAAS,CAAC,EAAEC,IAAII,IAAI,CAAC;kBACvF,EAAE,IAAI,CAACS,YAAY,CAACd,UAAUC,KAAKZ,OAAO;sBACtC,CAAC;YACrB,GACCQ,IAAI,CAAC,QAAQ;;;;;;;;;;;;;;;;;;;;;;;;;;;MA2BV,CAAC,CAACgF,IAAI;YACNC,YAAY,EAAE;YACdtB;QACF;IACF;AACF"}
481
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"sources":["../../../src/template/implementations/view_form.template.ts"],"sourcesContent":["import inflection from \"inflection\";\nimport { unique } from \"radashi\";\nimport { z } from \"zod\";\nimport { EntityManager, type EntityNamesRecord } from \"../../entity/entity-manager\";\nimport type { RenderingNode, TemplateKey, TemplateOptions } from \"../../types/types\";\nimport { getEnumInfoFromColName, getRelationPropFromColName } from \"../helpers\";\nimport type { RenderedTemplate } from \"../template\";\nimport { Template } from \"../template\";\n\nexport class Template__view_form extends Template {\n  constructor() {\n    super(\"view_form\");\n  }\n\n  getTargetAndPath(names: EntityNamesRecord) {\n    return {\n      target: \"web/src/routes/admin\",\n      path: `${names.fsPlural}/form.tsx`,\n    };\n  }\n\n  override getRequiredDictKeys(): string[] | null {\n    return [\"entity.create\", \"entity.edit\", \"common.backToList\", \"form.createdAt\", \"common.save\"];\n  }\n\n  wrapFC(body: string, label?: string): string {\n    return [\n      `<div className=\"space-y-2\">${label ? `\\n  <Label>${label}</Label>` : \"\"}`,\n      `  ${body}`,\n      `</div>`,\n    ].join(\"\\n\");\n  }\n  wrapFG(body: string, label?: string): string {\n    return this.wrapFC(body, label);\n  }\n\n  renderColumnImport(entityId: string, col: RenderingNode) {\n    if (col.renderType === \"enums\") {\n      const { id, targetEntityNames } = getEnumInfoFromColName(entityId, col.name);\n      const componentId = `${id}Select`;\n      return `import { ${componentId} } from \"@/components/${targetEntityNames.fs}/${componentId}\";`;\n    } else if (col.renderType === \"number-fk_id\") {\n      try {\n        const relProp = getRelationPropFromColName(entityId, col.name.replace(\"_id\", \"\"));\n        const targetNames = EntityManager.getNamesFromId(relProp.with);\n        const componentId = `${relProp.with}IdAsyncSelect`;\n        return `import { ${componentId} } from \"@/components/${targetNames.fs}/${componentId}\";`;\n      } catch {\n        return \"\";\n      }\n    } else {\n      throw new Error(`렌더 불가능한 임포트 ${col.name} ${col.renderType}`);\n    }\n  }\n\n  renderColumn(entityId: string, col: RenderingNode, names: EntityNamesRecord): string {\n    const regExpr = `{...register(\"${col.name}\")}`;\n    const placeholder = `{SD(\"entity.${entityId}.${col.name}\")}`;\n\n    switch (col.renderType) {\n      case \"string-plain\":\n        if (col.zodType instanceof z.ZodString && (col.zodType.maxLength ?? 0) <= 256) {\n          return `<Input className=\"h-8 text-xs bg-white\" placeholder=${placeholder} ${regExpr} />`;\n        } else {\n          return `<Textarea className=\"text-xs bg-white\" rows={4} placeholder=${placeholder} ${regExpr} />`;\n        }\n      case \"datetime\":\n        return `<DateInput\n                    className=\"h-8 text-xs bg-white\"\n                    ${regExpr}\n                  />`;\n      case \"number-id\":\n        return `<input type=\"hidden\" ${regExpr} />`;\n      case \"number-plain\":\n        return `<Input type=\"number\" className=\"h-8 text-xs bg-white\" placeholder=${placeholder} ${regExpr} />`;\n      case \"boolean\":\n        return `<Switch ${regExpr} />`;\n      case \"json-sonamufile\":\n        return `<FileInput\n                    uploadMode=\"lazy\"\n                    viewMode=\"image\"\n                    previewSize=\"md\"\n                    ${regExpr}\n                  />`;\n      case \"json-sonamufile-array\":\n        return `<FileInput\n                    multiple={true}\n                    uploadMode=\"lazy\"\n                    viewMode=\"image\"\n                    previewSize=\"md\"\n                    maxFiles={10}\n                    ${regExpr}\n                  />`;\n      case \"enums\":\n        try {\n          let enumId: string;\n          if (col.name === \"orderBy\") {\n            enumId = `${names.capital}${inflection.camelize(col.name)}Select`;\n          } else {\n            const { id } = getEnumInfoFromColName(entityId, col.name);\n            enumId = `${id}Select`;\n          }\n          return `<${enumId} ${regExpr} ${col.optional || col.nullable ? \"clearable\" : \"\"} />`;\n        } catch {\n          return `<Input className=\"h-8 text-xs bg-white\" ${regExpr} />`;\n        }\n      case \"number-fk_id\":\n        try {\n          const relProp = getRelationPropFromColName(entityId, col.name.replace(\"_id\", \"\"));\n          const fkId = `${relProp.with}IdAsyncSelect`;\n          return `<${fkId} subset=\"A\" ${regExpr} ${\n            col.optional || col.nullable ? \"clearable\" : \"\"\n          } className=\"h-8 text-xs\" />`;\n        } catch {\n          return `<Input type=\"number\" className=\"h-8 text-xs bg-white\" placeholder=${placeholder} ${regExpr} />`;\n        }\n      case \"array\":\n      case \"object\":\n        return `<Input className=\"h-8 text-xs bg-white\" placeholder=\"${col.name}\" ${regExpr} />`;\n      default:\n        return `<Input className=\"h-8 text-xs bg-white\" ${regExpr} />`;\n    }\n  }\n\n  resolveDefaultValue(columns: RenderingNode[]): object {\n    return columns.reduce(\n      (result, col) => {\n        if (col.optional) {\n          return result;\n        }\n\n        let value: unknown;\n        if (col.nullable === true) {\n          value = null;\n        } else if (col.zodType instanceof z.ZodNumber) {\n          value = 0;\n        } else if (col.zodType instanceof z.ZodEnum) {\n          value = Object.keys(col.zodType.enum)[0];\n        } else if (col.zodType instanceof z.ZodBoolean) {\n          value = false;\n        } else if (col.zodType instanceof z.core.$ZodString) {\n          // NOTE: z.ZodString으로 비교하면 z.url(), z.email() 등의 타입에서 문제가 생기므로 z.core.$ZodString으로 비교함\n          // FIXME: email이나 url 타입 등에 대한 처리가 필요함\n          if (col.renderType === \"string-datetime\") {\n            value = \"now()\";\n          } else {\n            value = \"\";\n          }\n        } else if (col.zodType instanceof z.ZodArray) {\n          value = [];\n        } else if (col.zodType instanceof z.ZodObject) {\n          value = {};\n        }\n\n        result[col.name] = value;\n        return result;\n      },\n      {} as { [key: string]: unknown },\n    );\n  }\n\n  async render({ entityId }: TemplateOptions[\"view_form\"]) {\n    const entity = EntityManager.get(entityId);\n    const names = EntityManager.getNamesFromId(entityId);\n\n    // SaveParams 타입을 로드하여 saveParamsNode 생성\n    const { loadTypes } = await import(\"../../syncer/module-loader\");\n    const loadedTypes = await loadTypes();\n    const SaveParamsZodType = loadedTypes[`${entityId}SaveParams`];\n\n    if (!SaveParamsZodType) {\n      throw new Error(`SaveParams for ${entityId} not found. Did you run 'sonamu sync'?`);\n    }\n\n    // Zod 타입을 RenderingNode로 변환\n    const { zodTypeToRenderingNode } = await import(\"../zod-converter\");\n    const saveParamsNode = zodTypeToRenderingNode(SaveParamsZodType);\n\n    const columns = ((saveParamsNode?.children ?? []) as RenderingNode[])\n      .filter((col) => col.name !== \"id\")\n      .map((col) => {\n        const propCandidate = entity.props.find((prop) => prop.name === col.name);\n        col.label = propCandidate?.desc ?? col.label;\n        return col;\n      });\n\n    const defaultValue = this.resolveDefaultValue(columns);\n\n    // 프리 템플릿\n    const preTemplates: RenderedTemplate[\"preTemplates\"] = (columns as RenderingNode[])\n      .filter((col) => {\n        if (col.name === \"id\") {\n          return false;\n        } else if (col.name.endsWith(\"_id\") || col.renderType === \"number-id\") {\n          try {\n            getRelationPropFromColName(entityId, col.name.replace(\"_id\", \"\"));\n            return true;\n          } catch {\n            return false;\n          }\n        } else if (col.renderType === \"enums\") {\n          try {\n            getEnumInfoFromColName(entityId, col.name);\n            return true;\n          } catch {\n            return false;\n          }\n        }\n        return false;\n      })\n      .map((col) => {\n        let key: TemplateKey;\n        let targetMdId = entityId;\n        let enumId: string | undefined;\n        if (col.renderType === \"enums\") {\n          key = \"view_enums_select\";\n          const { targetEntityNames: targetMDNames, id } = getEnumInfoFromColName(\n            entityId,\n            col.name,\n          );\n          targetMdId = targetMDNames.capital;\n          enumId = id;\n        } else {\n          key = \"view_id_async_select\";\n          const relProp = getRelationPropFromColName(entityId, col.name.replace(\"_id\", \"\"));\n          targetMdId = relProp.with;\n        }\n\n        return {\n          key: key as TemplateKey,\n          options: {\n            entityId: targetMdId,\n            node: col,\n            enumId,\n          },\n        };\n      })\n      .filter((preTemplate) => {\n        if (preTemplate.key === \"view_id_async_select\") {\n          try {\n            EntityManager.get(preTemplate.options.entityId);\n            return true;\n          } catch {\n            return false;\n          }\n        }\n        return true;\n      });\n\n    return {\n      ...this.getTargetAndPath(names),\n      body: `\nimport {\n  Button,\n  Card,\n  CardContent,\n  CardHeader,\n  CardTitle,\n  Input,${columns.some((col) => col.renderType === \"string-plain\" && col.zodType instanceof z.ZodString && (col.zodType.maxLength ?? 0) > 256) ? \"\\n  Textarea,\" : \"\"}${columns.some((col) => col.renderType === \"enums\") ? \"\\n  Select,\\n  SelectContent,\\n  SelectItem,\\n  SelectTrigger,\\n  SelectValue,\" : \"\"}${columns.some((col) => col.renderType === \"boolean\") ? \"\\n  Switch,\" : \"\"}${columns.some((col) => [\"json-sonamufile\", \"json-sonamufile-array\"].includes(col.renderType)) ? \"\\n  FileInput,\" : \"\"}${columns.some((col) => [\"string-datetime\", \"string-date\", \"datetime\"].includes(col.renderType)) ? \"\\n  DateInput,\" : \"\"}\n} from \"@sonamu-kit/react-components/components\";\nimport { useTypeForm } from \"@sonamu-kit/react-components/lib\";\nimport { useQueryClient } from \"@tanstack/react-query\";\nimport { createFileRoute, useRouter } from \"@tanstack/react-router\";\nimport { useEffect } from \"react\";\nimport { z } from \"zod\";\nimport { ${names.capital}Service } from \"@/services/services.generated\";\nimport type { ${names.capital}SubsetA } from \"@/services/sonamu.generated\";${\n        columns.filter((col) => col.renderType === \"enums\").length > 0\n          ? \"\\nimport { \" +\n            unique(\n              columns\n                .filter((col) => col.renderType === \"enums\")\n                .map((col) => {\n                  try {\n                    const { id } = getEnumInfoFromColName(entityId, col.name);\n                    return `${id}, ${id}Label`;\n                  } catch {\n                    return \"\";\n                  }\n                }),\n            )\n              .filter(Boolean)\n              .join(\", \") +\n            ' } from \"@/services/sonamu.generated\";'\n          : \"\"\n      }\nimport { defaultCatch } from \"@/services/sonamu.shared\";\nimport { ${names.capital}SaveParams } from \"@/services/${names.fs}/${names.fs}.types\";\n${unique(\n  columns\n    .filter((col) => [\"number-fk_id\", \"enums\"].includes(col.renderType))\n    .map((col) => {\n      return this.renderColumnImport(entityId, col);\n    }),\n).join(\"\\n\")}\nimport { SD } from \"@/i18n/sd.generated\";\n\nimport ArrowLeftIcon from \"~icons/lucide/arrow-left\";\nimport SaveIcon from \"~icons/lucide/save\";\nimport FormIcon from \"~icons/mdi/form-select\";\n\nconst formSearchSchema = z.object({\n  id: z.number().optional(),\n});\n\nexport const Route = createFileRoute(\"/admin/${names.fsPlural}/form\")({\n  validateSearch: formSearchSchema,\n  component: ${names.capitalPlural}FormPage,\n});\n\nfunction ${names.capitalPlural}FormPage() {\n  const { id } = Route.useSearch();\n  return <${names.capitalPlural}Form id={id} />;\n}\n\ntype ${names.capitalPlural}FormProps = {\n  id?: number;\n  mode?: \"page\" | \"modal\";\n};\n\nexport function ${names.capitalPlural}Form({ id, mode }: ${names.capitalPlural}FormProps) {\n  const router = useRouter();\n  const queryClient = useQueryClient();\n\n  const { form, setForm, register${columns.some((col) => [\"json-sonamufile\", \"json-sonamufile-array\"].includes(col.renderType)) ? \", submit\" : \"\"} } = useTypeForm(${names.capital}SaveParams, ${JSON.stringify(defaultValue).replace(/\"now\\(\\)\"/g, '\"\"')});\n${(() => {\n  const hasDatetime = columns.some((col) => col.renderType === \"string-datetime\");\n  const hasDate = columns.some((col) => col.renderType === \"string-date\");\n  if (!hasDatetime && !hasDate) return \"\";\n\n  let helpers = \"\\n\";\n  if (hasDatetime) {\n    helpers += `  // datetime-local 형식으로 변환 (YYYY-MM-DDTHH:MM)\n  const toDatetimeLocalString = (date: Date | string | null | undefined): string => {\n    if (!date) return \"\";\n    const d = typeof date === \"string\" ? new Date(date) : date;\n    return d.toISOString().slice(0, 16);\n  };\n\n  // datetime-local 문자열을 Date로 변환\n  const fromDatetimeLocalString = (value: string): Date | null => {\n    if (!value) return null;\n    return new Date(value);\n  };\n`;\n  }\n  if (hasDate) {\n    helpers += `  // date 형식으로 변환 (YYYY-MM-DD)\n  const toDateString = (date: Date | string | null | undefined): string => {\n    if (!date) return \"\";\n    const d = typeof date === \"string\" ? new Date(date) : date;\n    return d.toISOString().split(\"T\")[0];\n  };\n\n  // date 문자열을 Date로 변환\n  const fromDateString = (value: string): Date | null => {\n    if (!value) return null;\n    return new Date(value);\n  };\n`;\n  }\n  return helpers;\n})()}\n  useEffect(() => {\n    if (id) {\n      ${names.capital}Service.get${names.capital}(\"A\", id).then((row) => {\n        setForm((prevForm) => ({\n          ...prevForm,\n          ...row,${(() => {\n            // relation 필드들을 찾아서 변환 코드 생성\n            const relationFields = columns\n              .filter((col) => col.renderType === \"number-fk_id\")\n              .map((col) => {\n                const relationName = col.name.replace(/_id$/, \"\");\n                if (col.nullable) {\n                  return `\\n          ${col.name}: row.${relationName}?.id ?? null,`;\n                } else {\n                  return `\\n          ${col.name}: row.${relationName}?.id,`;\n                }\n              })\n              .join(\"\");\n            return relationFields;\n          })()}\n        }));\n      });\n    }\n  }, [id, setForm]);\n\n  const saveMutation = ${names.capital}Service.useSaveMutation();\n  const handleSubmit = ${\n    columns.some((col) => [\"json-sonamufile\", \"json-sonamufile-array\"].includes(col.renderType))\n      ? `submit(async (form) => {\n    saveMutation.mutate(\n      { spa: [form] },\n      {\n        onSuccess: () => {\n          queryClient.invalidateQueries({\n            queryKey: [\"${names.capital}\"],\n          });\n\n          if (mode === \"modal\") {\n            // modal mode\n          } else {\n            router.navigate({ to: \"/admin/${names.fsPlural}\" });\n          }\n        },\n        onError: defaultCatch,\n      },\n    );\n  })`\n      : `() => {\n    saveMutation.mutate(\n      { spa: [form] },\n      {\n        onSuccess: () => {\n          queryClient.invalidateQueries({\n            queryKey: [\"${names.capital}\"],\n          });\n\n          if (mode === \"modal\") {\n            // modal mode\n          } else {\n            router.navigate({ to: \"/admin/${names.fsPlural}\" });\n          }\n        },\n        onError: defaultCatch,\n      },\n    );\n  }`\n  };\n\n  const PAGE = {\n    title: id ? SD(\"entity.edit\")(SD(\"entity.${entityId}\"), id) : SD(\"entity.create\")(SD(\"entity.${entityId}\")),\n  };\n\n  return (\n    <div className=\"flex-1 overflow-auto\">\n      <div className=\"max-w-[1800px] mx-auto p-8\">\n        <div className=\"space-y-6 mb-8\">\n          {/* Header */}\n          <div className=\"flex items-center justify-between\">\n            <div className=\"flex items-center gap-2\">\n              <FormIcon className=\"h-5 w-5\" />\n              <span className=\"text-lg font-semibold h-5\">{PAGE.title}</span>\n            </div>\n            {mode !== \"modal\" && (\n              <Button\n                variant=\"outline\"\n                onClick={() => router.navigate({ to: \"/admin/${names.fsPlural}\" })}\n                icon={<ArrowLeftIcon />}\n              >\n                {SD(\"common.backToList\")}\n              </Button>\n            )}\n          </div>\n\n          {/* Form Card */}\n          <Card className=\"border-border/40 bg-gray-50 shadow-sm\">\n            <CardHeader className=\"px-4 border-b border-gray-200 flex items-center\">\n              <CardTitle className=\"text-sm font-medium leading-none m-0\">\n                {PAGE.title}\n              </CardTitle>\n            </CardHeader>\n            <CardContent className=\"p-6\">\n              <div className=\"space-y-6\">\n${columns\n  .filter((col) => col.name !== \"created_at\")\n  .map((col) => {\n    const label = (() => {\n      if (col.label.endsWith(\"Id\")) {\n        try {\n          const entity = EntityManager.get(col.label.replace(\"Id\", \"\"));\n          return entity.title ?? col.label;\n        } catch {\n          return col.label;\n        }\n      }\n      return col.label;\n    })();\n    return `                {/* ${label} */}\n                <div className=\"space-y-2\">\n                  <label className=\"block text-xs mb-1 text-gray-600\">{SD(\"entity.${entityId}.${col.name}\")}</label>\n                  ${this.renderColumn(entityId, col, names)}\n                </div>`;\n  })\n  .join(\"\\n\\n\")}\n\n                {/* Save Button */}\n                <div className=\"flex items-center justify-between pt-4\">\n                  {form.id && form.created_at && (\n                    <div className=\"flex items-center\">\n                      <label className=\"mr-2 text-xs text-gray-600\">{SD(\"form.createdAt\")}:</label>\n                      <span className=\"text-xs text-gray-600\">\n                        {String(form.created_at)}\n                      </span>\n                    </div>\n                  )}\n                  <Button\n                    onClick={handleSubmit}\n                    icon={<SaveIcon />}\n                  >\n                    {SD(\"common.save\")}\n                  </Button>\n                </div>\n              </div>\n            </CardContent>\n          </Card>\n        </div>\n      </div>\n    </div>\n  );\n}\n      `.trim(),\n      importKeys: [],\n      preTemplates,\n    };\n  }\n}\n"],"names":["inflection","unique","z","EntityManager","getEnumInfoFromColName","getRelationPropFromColName","Template","Template__view_form","getTargetAndPath","names","target","path","fsPlural","getRequiredDictKeys","wrapFC","body","label","join","wrapFG","renderColumnImport","entityId","col","renderType","id","targetEntityNames","name","componentId","fs","relProp","replace","targetNames","getNamesFromId","with","Error","renderColumn","regExpr","placeholder","zodType","ZodString","maxLength","enumId","capital","camelize","optional","nullable","fkId","resolveDefaultValue","columns","reduce","result","value","ZodNumber","ZodEnum","Object","keys","enum","ZodBoolean","core","$ZodString","ZodArray","ZodObject","render","entity","get","loadTypes","loadedTypes","SaveParamsZodType","zodTypeToRenderingNode","saveParamsNode","children","filter","map","propCandidate","props","find","prop","desc","defaultValue","preTemplates","endsWith","key","targetMdId","targetMDNames","options","node","preTemplate","some","includes","length","Boolean","capitalPlural","JSON","stringify","hasDatetime","hasDate","helpers","relationFields","relationName","title","trim","importKeys"],"mappings":"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"}