Kaizhi Zheng, Xiaotong Chen, Xuehai He, Jing Gu, Linjie Li, Zhengyuan Yang, Kevin Lin, Jianfeng Wang, Lijuan Wang, Xin Eric Wang
This repository contains the official implementation of the paper: EditRoom: LLM-parameterized Graph Diffusion for Composable 3D Room Layout Editing, which is accepted by ICLR 2025. EditRoom is a graph diffusion-based generative model, which can manipulate 3D Room by using natural lanauge.
Feel free to contact me ([email protected]) or open an issue if you have any questions or suggestions.
Clone our repo and create a new python environment.
git clone https://github.com/eric-ai-lab/EditRoom.git
cd EditRoom
conda create -n editroom python=3.10
conda activate editroom
pip install -r requirements.txtDownload the Blender software for visualization.
mkdir -p blender
cd blender
wget https://download.blender.org/release/Blender3.3/blender-3.3.1-linux-x64.tar.xz
tar -xvf blender-3.3.1-linux-x64.tar.xz
rm blender-3.3.1-linux-x64.tar.xzDataset used in EditRoom is based on 3D-FORNT and 3D-FUTURE. Please refer to the instructions provided in their official website to download the original dataset. Based on InstructScene, we write an automatic dataset generator to generate scene editing pairs. We provided preprocessed datasets on HuggingFace for start.
First, downloading the preprocessed datasets. They will be downloaded under datasets folder by default. If you want to change to another directory, please remind to chage environment viariable EDITROOM_DATA_FOLDER.
export EDITROOM_DATA_FOLDER="./datasets"
PYTHONPATH=. python3 tools/download_dataset.pyNote: After executing this command, the following files will be automatically unzipped:
datasets/objfeat_vqvae.zipwill be unzipped todatasets/data2/kzheng31/EditRoom_Public/objfeat_vqvaedatasets/preprocess.zipwill be unzipped todatasets/data2/zhengkz/3D_datasets/preprocess
However, both objfeat_vqvae and preprocess folders need to be directly under the datasets folder to comply with the required directory structure. Please run the following commands from the root directory to move them to the correct locations:
# Move objfeat_vqvae folder to datasets/
mv datasets/data2/kzheng31/EditRoom_Public/editroom_dataset/objfeat_vqvae datasets/
# Move preprocess folder to datasets/
mv datasets/data2/zhengkz/3D_datasets/preprocess datasets/
# Verify the folders are in the correct locations
ls -la datasets/objfeat_vqvae
ls -la datasets/preprocess
Then, please refer to tools/README.md for more details.
After dataset preparation and setup, your directory structure should look like this:
EditRoom
├── assets
│ └── new_teaser-1.png
├── blender
│ └── blender-3.3.1-linux-x64
│ ├── 3.3
│ ├── blender
│ ├── blender.desktop
│ ├── blender-softwaregl
│ ├── blender.svg
│ ├── blender-symbolic.svg
│ ├── blender-thumbnailer
│ ├── copyright.txt
│ ├── lib
│ ├── license
│ └── readme.html
├── configs
│ ├── bedroom_sg2sc_diffusion.yaml
│ ├── bedroom_sg_diffusion.yaml
│ ├── bedroom_threed_front_splits.csv
│ ├── black_list.txt
│ ├── diningroom_sg2sc_diffusion.yaml
│ ├── diningroom_sg_diffusion.yaml
│ ├── diningroom_threed_front_splits.csv
│ ├── floor_plan_texture_images
│ │ └── floor_00003.jpg
│ ├── floor_plan_texture_images_references
│ ├── invalid_threed_front_rooms.txt
│ ├── livingroom_sg2sc_diffusion.yaml
│ ├── livingroom_sg_diffusion.yaml
│ └── livingroom_threed_front_splits.csv
├── constants.py
├── datasets
│ ├── 3D-FRONT
│ │ ├── 3D-FRONT
│ │ ├── 3D-FRONT-readme.md
│ │ ├── 3D-FRONT-texture
│ │ ├── 3D-FUTURE-model
│ │ └── threed_front.pkl
│ ├── 3D-FRONT.zip
│ ├── data2
│ │ ├── kzheng31
│ │ └── zhengkz
│ ├── editroom_dataset
│ │ ├── threed_front_bedroom
│ │ ├── threed_front_diningroom
│ │ └── threed_front_livingroom
│ ├── objfeat_vqvae
│ │ ├── objfeat_bounds.pkl
│ │ └── threedfront_objfeat_vqvae_epoch_01999.pth
│ ├── objfeat_vqvae.zip
│ ├── preprocess
│ │ ├── openshape_vitg14_indexs
│ │ ├── openshape_vitg14_indexs.tar
│ │ ├── openshape_vitg14_recon
│ │ ├── openshape_vitg14_recon.tar
│ │ ├── room_relations
│ │ └── room_relations.tar
│ └── preprocess.zip
├── extract_llm_plans.py
├── README.md
├── requirements.txt
├── run.sh
├── src
│ ├── data
│ ├── eval_edit.py
│ ├── infer.py
│ ├── models
│ ├── train_edit.py
│ └── utils
├── tools
└── weights (if you've downloaded ckpts and data pkl files from HF)
├── threedfront_bedroom_sg2sc_model-epoch=03-val_loss=0.171.ckpt
├── threedfront_bedroom_sg2sc_model-epoch=04-val_loss=0.146.ckpt
├── threedfront_bedroom_sg2sc_model-epoch=269-val_loss=0.018.ckpt
├── threedfront_bedroom_sg2sc_model-epoch=273-val_loss=0.018.ckpt
├── threedfront_bedroom_sg_model-epoch=213-val_loss=0.107.ckpt
├── threedfront_bedroom_sg_model-epoch=87-val_loss=0.111.ckpt
├── threedfront_bedroom_test_data.pkl
├── threed_front_bedroom_test_eval_tmp.pkl
├── threedfront_bedroom_train_data.pkl
We will first train scene graph to scene layout generator:
PYTHONPATH=. python3 src/train_edit.py --config_file configs/bedroom_sg2sc_diffusion.yaml --output_directory ./weights --with_wandb_loggerThen, we will train scene graph generator:
PYTHONPATH=. python3 src/train_edit.py --config_file configs/bedroom_sg_diffusion.yaml --output_directory ./weights --with_wandb_loggerYou can change config_file to other room types. All config files are under configs folder.
To run the evaluation, we need both SG_WEIGHT and SG2SC_WEIGHT. Those weights should be found under output_directory for training.
If you runing on a headless server, please using Xvfb to create virtual screens for pyrender.
tmux new -s v_screen #Opening another terminal at the backend
sudo Xvfb :99 -screen 0 1960x1024x24For evaluation, please run:
export DISPLAY=":99" #For headless server
PYTHONPATH=. python3 src/eval_edit.py --sg_config_file configs/bedroom_sg_diffusion.yaml \
--sg2sc_config_file configs/bedroom_sg2sc_diffusion.yaml \
--output_directory ./weights \
--sg_weight_file SG_WEIGHT \
--sg2sc_weight_file SG2SC_WEIGHT \
--llm_plan_path LLM_PLANLLM_PLAN should be created during dataset generation. Please refer to tools/README.md for more details.
You can also using tempalte commands for evaluation.
export DISPLAY=":99" #For headless server
PYTHONPATH=. python3 src/eval_edit.py --sg_config_file configs/bedroom_sg_diffusion.yaml \
--sg2sc_config_file configs/bedroom_sg2sc_diffusion.yaml \
--output_directory ./weights \
--sg_weight_file SG_WEIGHT \
--sg2sc_weight_file SG2SC_WEIGHTBefore running inference, you need to download the trained model checkpoints from our private Hugging Face repository.
First, login to Hugging Face CLI (this will prompt for your access token):
huggingface-cli loginEnter your Hugging Face access token when prompted. You can obtain an access token from Hugging Face Settings with read access to private repositories.
Then download the trained checkpoints:
python3 weights_download.pyThis will download trained checkpoints from Hugging Face to the ./weights directory for running inference.
This guide shows how to use the infer.py script for interactive 3D scene editing. infer.py can be run directly after weights_download.py and edit pair creation by edit_data_generator
| Argument | Description | Example |
|---|---|---|
--source_scene_id |
Source scene UID (without .pkl extension) | 0d7be408-9e3d-4f68-8422-5aa2069ccdb2_MasterBedroom-27127 |
--room_type |
Room type (bedroom/livingroom/diningroom) | bedroom |
--sg_config_file |
Path to scene graph config file | ./configs/bedroom_sg_diffusion.yaml |
--sg2sc_config_file |
Path to scene graph to scene config file | ./configs/bedroom_sg2sc_diffusion.yaml |
--sg_weight_file |
Path to scene graph model weights | /path/to/sg_model.ckpt |
--sg2sc_weight_file |
Path to scene graph to scene model weights | /path/to/sg2sc_model.ckpt |
--output_directory |
Output directory for results | single_scene_results (default) |
--no_edit |
Only visualize original scene without editing | N/A (flag) |
--seed |
Random seed for reproducibility | 42 (default) |
--quiet |
Suppress verbose output and debug messages | N/A (flag) |
Use this mode to render and save the original scene without any editing:
PYTHONPATH=. python src/infer.py \
--source_scene_id [SRC_SCENE_ID] \
--room_type [bedroom/livingroom/diningroom] \
--sg_config_file ./configs/bedroom_sg_diffusion.yaml \
--sg2sc_config_file ./configs/bedroom_sg2sc_diffusion.yaml \
--sg_weight_file [SG_WT_FILE_PATH] \
--sg2sc_weight_file [SG2SC_WT_FILE_PATH] \
--no_editUse this mode to edit scenes with natural language commands:
PYTHONPATH=. python src/infer.py \
--source_scene_id [SRC_SCENE_ID] \
--room_type [bedroom/livingroom/diningroom] \
--sg_config_file ./configs/bedroom_sg_diffusion.yaml \
--sg2sc_config_file ./configs/bedroom_sg2sc_diffusion.yaml \
--sg_weight_file [SG_WT_FILE_PATH] \
--sg2sc_weight_file [SG2SC_WT_FILE_PATH] \The script will prompt you to enter editing commands like:
- "add a wardrobe next to the bed"
- "move the chair to the left"
- "rotate the desk 90 degrees"
Use this mode to reduce verbose output during editing:
PYTHONPATH=. python src/infer.py \
--source_scene_id [SRC_SCENE_ID] \
--room_type [bedroom/livingroom/diningroom] \
--sg_config_file ./configs/bedroom_sg_diffusion.yaml \
--sg2sc_config_file ./configs/bedroom_sg2sc_diffusion.yaml \
--sg_weight_file [SG_WT_FILE_PATH] \
--sg2sc_weight_file [SG2SC_WT_FILE_PATH] \
--quietResults are saved in timestamped folders under the output directory:
- Original scene visualization:
{output_dir}/{scene_id}_{timestamp}/original/ - Edited scene results:
{output_dir}/{scene_id}_{timestamp}/generate/ - Multi-step editing:
{output_dir}/{scene_id}_{timestamp}/edit_1/,edit_2/, etc.
- Valid OpenAI API key (set as
OPENAI_API_KEYenvironment variable insideconstants.py) - Preprocessed 3D-FRONT dataset with VQ features
- Trained model weights for scene graph and scene generation
- Blender installation as per EditRoom author guidelines (for rendering)
- SRC_SCENE_ID must be present in either train or test folders under
./datasets/editroom_dataset/threed_front_{room_type}/
We would like to thank the authors of ATISS, DiffuScene, OpenShape, NAP, CLIPLayout and InstructScene for their great work and generously providing source codes, which inspired our work and helped us a lot in the implementation.
If you find our work helpful, please consider citing:
@inproceedings{
zheng2025editroom,
title={EditRoom: {LLM}-parameterized Graph Diffusion for Composable 3D Room Layout Editing},
author={Kaizhi Zheng and Xiaotong Chen and Xuehai He and Jing Gu and Linjie Li and Zhengyuan Yang and Kevin Lin and Jianfeng Wang and Lijuan Wang and Xin Eric Wang},
booktitle={The Thirteenth International Conference on Learning Representations},
year={2025}
}