MoleditPy 2.4.0__py3-none-any.whl → 2.4.2__py3-none-any.whl

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.
@@ -16,7 +16,7 @@ from PyQt6.QtGui import QFont, QColor
16
16
  from rdkit import Chem
17
17
 
18
18
  #Version
19
- VERSION = '2.4.0'
19
+ VERSION = '2.4.2'
20
20
 
21
21
  ATOM_RADIUS = 18
22
22
  BOND_OFFSET = 3.5
@@ -500,6 +500,10 @@ class MainWindow(QMainWindow):
500
500
  # --- MOVED TO main_window_export.py ---
501
501
  return self.main_window_export.export_2d_png()
502
502
 
503
+ def export_2d_svg(self):
504
+ # --- MOVED TO main_window_export.py ---
505
+ return self.main_window_export.export_2d_svg()
506
+
503
507
  def export_3d_png(self):
504
508
  # --- MOVED TO main_window_export.py ---
505
509
  return self.main_window_export.export_3d_png()
@@ -528,6 +532,10 @@ class MainWindow(QMainWindow):
528
532
  # --- MOVED TO main_window_edit_actions.py ---
529
533
  return self.main_window_edit_actions.open_rotate_2d_dialog()
530
534
 
535
+ def rotate_molecule_2d(self, angle_degrees):
536
+ # --- MOVED TO main_window_edit_actions.py ---
537
+ return self.main_window_edit_actions.rotate_molecule_2d(angle_degrees)
538
+
531
539
  def draw_molecule_3d(self, mol):
532
540
  # --- MOVED TO main_window_view_3d.py ---
533
541
  return self.main_window_view_3d.draw_molecule_3d(mol)
@@ -618,18 +618,15 @@ class MainWindowEditActions(object):
618
618
  self.rotate_molecule_2d(angle)
619
619
 
620
620
  def rotate_molecule_2d(self, angle_degrees):
621
- """2D分子を指定角度回転させる(選択範囲があればそれのみ、なければ全体)"""
621
+ """2D分子を指定角度回転させる(選択範囲があればそれのみ)"""
622
622
  try:
623
623
  # Determine target atoms
624
624
  selected_items = self.scene.selectedItems()
625
625
  target_atoms = [item for item in selected_items if isinstance(item, AtomItem)]
626
626
 
627
- # If no selection, rotate everything
627
+ # If no selection, do not rotate
628
628
  if not target_atoms:
629
- target_atoms = [data['item'] for data in self.data.atoms.values() if data.get('item') and not sip_isdeleted_safe(data['item'])]
630
-
631
- if not target_atoms:
632
- self.statusBar().showMessage("No atoms to rotate.")
629
+ self.statusBar().showMessage("No atoms selected to rotate.")
633
630
  return
634
631
 
635
632
  # Calculate Center
@@ -663,6 +660,8 @@ class MainWindowEditActions(object):
663
660
  self.push_undo_state()
664
661
  self.statusBar().showMessage(f"Rotated {len(target_atoms)} atoms by {angle_degrees} degrees.")
665
662
  self.scene.update()
663
+ # Force full redraw as requested
664
+ self.scene.update_all_items()
666
665
 
667
666
  except Exception as e:
668
667
  print(f"Error rotating molecule: {e}")
@@ -37,9 +37,11 @@ from PyQt6.QtGui import (
37
37
  QBrush, QColor, QPainter, QImage
38
38
  )
39
39
 
40
+ from PyQt6.QtSvg import QSvgGenerator
41
+
40
42
 
41
43
  from PyQt6.QtCore import (
42
- Qt, QRectF
44
+ Qt, QRectF, QSize
43
45
  )
44
46
 
45
47
  import pyvista as pv
@@ -753,6 +755,118 @@ class MainWindowExport(object):
753
755
 
754
756
 
755
757
 
758
+ def export_2d_svg(self):
759
+ """2D drawingをSVGとしてエクスポート"""
760
+ if not self.data.atoms:
761
+ self.statusBar().showMessage("Nothing to export.")
762
+ return
763
+
764
+ # default filename
765
+ default_name = "untitled-2d"
766
+ try:
767
+ if self.current_file_path:
768
+ base = os.path.basename(self.current_file_path)
769
+ name = os.path.splitext(base)[0]
770
+ default_name = f"{name}-2d"
771
+ except Exception:
772
+ default_name = "untitled-2d"
773
+
774
+ # prefer same directory
775
+ default_path = default_name
776
+ try:
777
+ if self.current_file_path:
778
+ default_path = os.path.join(os.path.dirname(self.current_file_path), default_name)
779
+ except Exception:
780
+ default_path = default_name
781
+
782
+ filePath, _ = QFileDialog.getSaveFileName(self, "Export 2D as SVG", default_path, "SVG Files (*.svg)")
783
+ if not filePath:
784
+ return
785
+
786
+ if not (filePath.lower().endswith(".svg")):
787
+ filePath += ".svg"
788
+
789
+ # Ask about transparency
790
+ reply = QMessageBox.question(self, 'Choose Background',
791
+ 'Do you want a transparent background?\n(Choose "No" to use the current background color)',
792
+ QMessageBox.StandardButton.Yes | QMessageBox.StandardButton.No | QMessageBox.StandardButton.Cancel,
793
+ QMessageBox.StandardButton.Yes)
794
+
795
+ if reply == QMessageBox.StandardButton.Cancel:
796
+ self.statusBar().showMessage("Export cancelled.", 2000)
797
+ return
798
+
799
+ is_transparent = (reply == QMessageBox.StandardButton.Yes)
800
+
801
+ try:
802
+ # 1. Hide non-molecular items if needed (optional, keeping consistent with PNG export)
803
+ items_to_restore = {}
804
+ original_background = self.scene.backgroundBrush()
805
+
806
+ all_items = list(self.scene.items())
807
+ for item in all_items:
808
+ is_mol_part = isinstance(item, (AtomItem, BondItem))
809
+ if not (is_mol_part and item.isVisible()):
810
+ # Keep measurement items visible if they are part of the scene?
811
+ # For now, let's stick to hiding everything that isn't atom/bond,
812
+ # similar to png export logic, or we can decide to export everything visible.
813
+ # The PNG export hides non-atom/bond items. Let's follow that for consistency.
814
+ items_to_restore[item] = item.isVisible()
815
+ item.hide()
816
+
817
+ # 2. Calculate bounds
818
+ molecule_bounds = QRectF()
819
+ for item in self.scene.items():
820
+ if isinstance(item, (AtomItem, BondItem)) and item.isVisible():
821
+ molecule_bounds = molecule_bounds.united(item.sceneBoundingRect())
822
+
823
+ if molecule_bounds.isEmpty() or not molecule_bounds.isValid():
824
+ self.statusBar().showMessage("Error: Could not determine molecule bounds for export.")
825
+ # Restore
826
+ for item, was_visible in items_to_restore.items():
827
+ item.setVisible(was_visible)
828
+ return
829
+
830
+ if is_transparent:
831
+ self.scene.setBackgroundBrush(QBrush(Qt.BrushStyle.NoBrush))
832
+
833
+ # Margin
834
+ rect_to_render = molecule_bounds.adjusted(-20, -20, 20, 20)
835
+
836
+ width = int(rect_to_render.width())
837
+ height = int(rect_to_render.height())
838
+
839
+ # 3. Setup QSvgGenerator
840
+ generator = QSvgGenerator()
841
+ generator.setFileName(filePath)
842
+ generator.setSize(QSize(width, height))
843
+ generator.setViewBox(rect_to_render)
844
+ generator.setTitle("MoleditPy Molecule")
845
+
846
+ # 4. Render
847
+ painter = QPainter()
848
+ painter.begin(generator)
849
+ try:
850
+ self.scene.render(painter, rect_to_render, rect_to_render)
851
+ finally:
852
+ painter.end()
853
+
854
+ self.statusBar().showMessage(f"2D view exported to {filePath}")
855
+
856
+ except Exception as e:
857
+ self.statusBar().showMessage(f"An unexpected error occurred during SVG export: {e}")
858
+
859
+ finally:
860
+ # Restore
861
+ for item, was_visible in items_to_restore.items():
862
+ item.setVisible(was_visible)
863
+ if 'original_background' in locals():
864
+ self.scene.setBackgroundBrush(original_background)
865
+ if self.view_2d:
866
+ self.view_2d.viewport().update()
867
+
868
+
869
+
756
870
  def export_3d_png(self):
757
871
  if not self.current_mol:
758
872
  self.statusBar().showMessage("No 3D molecule to export.", 2000)
@@ -920,6 +920,10 @@ class MainWindowMainInit(object):
920
920
  export_2d_png_action = QAction("PNG Image...", self)
921
921
  export_2d_png_action.triggered.connect(self.export_2d_png)
922
922
  export_2d_menu.addAction(export_2d_png_action)
923
+
924
+ export_2d_svg_action = QAction("SVG Image...", self)
925
+ export_2d_svg_action.triggered.connect(self.export_2d_svg)
926
+ export_2d_menu.addAction(export_2d_svg_action)
923
927
 
924
928
  # 3D エクスポート
925
929
  export_3d_menu = export_menu.addMenu("3D Formats")
@@ -1597,6 +1601,15 @@ class MainWindowMainInit(object):
1597
1601
  except Exception:
1598
1602
  pass
1599
1603
  # Update 2D scene styling to reflect default CPK colors
1604
+ try:
1605
+ # Reset 2D background specifically
1606
+ if hasattr(self, 'scene') and self.scene:
1607
+ bg_c = self.settings.get('background_color_2d', '#FFFFFF')
1608
+ self.scene.setBackgroundBrush(QBrush(QColor(bg_c)))
1609
+
1610
+ self.update_cpk_colors_from_settings()
1611
+ except Exception:
1612
+ pass
1600
1613
  try:
1601
1614
  if hasattr(self, 'scene') and self.scene:
1602
1615
  for it in list(self.scene.items()):
@@ -116,6 +116,20 @@ class MoleculeScene(QGraphicsScene):
116
116
  self.reinitialize_items()
117
117
 
118
118
 
119
+ def update_connected_bonds(self, atoms):
120
+ """指定された原子リストに接続する全ての結合の位置を更新する"""
121
+ bonds_to_update = set()
122
+ for atom in atoms:
123
+ if hasattr(atom, 'bonds'):
124
+ bonds_to_update.update(atom.bonds)
125
+
126
+ for bond in bonds_to_update:
127
+ try:
128
+ if not sip_isdeleted_safe(bond):
129
+ bond.update_position()
130
+ except Exception:
131
+ continue
132
+
119
133
  def update_all_items(self):
120
134
  """全てのアイテムを強制的に再描画する"""
121
135
  for item in self.items():
@@ -1,6 +1,6 @@
1
1
  Metadata-Version: 2.4
2
2
  Name: MoleditPy
3
- Version: 2.4.0
3
+ Version: 2.4.2
4
4
  Summary: A cross-platform, simple, and intuitive molecular structure editor built in Python. It allows 2D molecular drawing and 3D structure visualization. It supports exporting structure files for input to DFT calculation software.
5
5
  Author-email: HiroYokoyama <titech.yoko.hiro@gmail.com>
6
6
  License: GNU GENERAL PUBLIC LICENSE
@@ -12,20 +12,20 @@ moleditpy/modules/bond_item.py,sha256=hOwga7DcxZf8zxwZr8F7viTNgEi_9wftKFgSpROUJs
12
12
  moleditpy/modules/bond_length_dialog.py,sha256=6bFPGssnqlgINuqpxLv-OhjMH3_hspnaH8QtorAyu2M,14782
13
13
  moleditpy/modules/calculation_worker.py,sha256=KiGQY7i-QCQofEoE0r65KoQgpEGFcbhmxWv6egfkUdc,42324
14
14
  moleditpy/modules/color_settings_dialog.py,sha256=Ow44BhCOLo0AFb6klO001k6B4drOgKX9DeNBQhZLp5o,15474
15
- moleditpy/modules/constants.py,sha256=hrpODQcypcLqXV8--0NiGgWRyWgEKcS6ju3Q-2RBY1w,4702
15
+ moleditpy/modules/constants.py,sha256=0jpJ6XvyBlso4hWUR9UqslTG-g8maR7_yipw9e9HoRk,4702
16
16
  moleditpy/modules/constrained_optimization_dialog.py,sha256=REsk4ePsqNmAGPMTS_jckeM7jexrU3krwun8sKqKUCs,30062
17
17
  moleditpy/modules/custom_interactor_style.py,sha256=LDNODMJoNHGe1AUSrvqv6PdeJm-hpPmSpWINppnJLt0,38942
18
18
  moleditpy/modules/custom_qt_interactor.py,sha256=vCZsDfRO-FtphD5cTP7Ps-5rpHZMIGloaoe6EaKzrsw,4139
19
19
  moleditpy/modules/dialog3_d_picking_mixin.py,sha256=z4udbkiX9PYmIGazPXsbftkk_oRRwZhcvlCqbyJzr24,6493
20
20
  moleditpy/modules/dihedral_dialog.py,sha256=bOTDO6-b74vEDn_z6OyuBr5cRz3RnRj83PiaEBUyWJA,18002
21
- moleditpy/modules/main_window.py,sha256=s5O-9B2MDrRg1OrXtq7rVtnqaQ9arWItGZ4Wgbn87SQ,36504
21
+ moleditpy/modules/main_window.py,sha256=w52L0F7V3b0BiNhQImbulZ9N8cRKolYy06szfoMOIcg,36830
22
22
  moleditpy/modules/main_window_app_state.py,sha256=8YDcGNCSpLTO1NGL9tEvNkXpUcS7JW-uK7TdUGvEqnk,35189
23
23
  moleditpy/modules/main_window_compute.py,sha256=ipIkhH_DONXDnPzh7xeym9X-Yfx8EhsvXYOdyxsAj4c,53347
24
24
  moleditpy/modules/main_window_dialog_manager.py,sha256=QR96LqHAPSOShXbc9cK-Ffq8a16JrXAoMKB0pHjESrQ,20072
25
25
  moleditpy/modules/main_window_edit_3d.py,sha256=CUArB5wcsgq1C7LygAEC6URlbnn4RhRYDa5n-Y-etWI,19731
26
- moleditpy/modules/main_window_edit_actions.py,sha256=jjMBwCzwflEHHvf_zh_RBcup9ceR_H4dDEvdRrb6wno,69009
27
- moleditpy/modules/main_window_export.py,sha256=YVrcFaY9xffFJ6eEH6Omh8Zif8KIs6_VDW_EAzN97AA,38218
28
- moleditpy/modules/main_window_main_init.py,sha256=Qkpb2AvMADmQmdZ1190ufQ3OWwvcQTwu440riyBn9vg,94148
26
+ moleditpy/modules/main_window_edit_actions.py,sha256=9CLyhfxLWG-zrqJXw_J5fwKzd3U3M5vKmQsy5qTFSJ8,68889
27
+ moleditpy/modules/main_window_export.py,sha256=_Vd7MeP_xaLWDYDm3-ZIZiPuAXhP-AvrQbi7Yx3Jy3c,43020
28
+ moleditpy/modules/main_window_main_init.py,sha256=t-um1lHvXo_tPXSkjx5Y0mjZlrKUhZbmb529e-7ffNU,94781
29
29
  moleditpy/modules/main_window_molecular_parsers.py,sha256=KR6vzuqc3nutOcorpYr0QOyX3MFBcxTwDhZX96VgJ9Q,48291
30
30
  moleditpy/modules/main_window_project_io.py,sha256=TWwtuKDuvgcvPZ9IGmW8r1EJJOrgxrIJRnxe_f4C1oM,17149
31
31
  moleditpy/modules/main_window_string_importers.py,sha256=v47wOd4RtjKYcF-aLP-mogGGdYTpTEo3dDyAu79_5MM,10782
@@ -34,7 +34,7 @@ moleditpy/modules/main_window_view_3d.py,sha256=CxZxyJHl2isF7KtyVWSI9f8LVbvdZM5H
34
34
  moleditpy/modules/main_window_view_loaders.py,sha256=gklTMo27QnyJ8Gd0ampPdbm9d0Gi-oHWkIqQuGADHmI,14352
35
35
  moleditpy/modules/mirror_dialog.py,sha256=c3v4qY6R4FAljzk4EPaDjL9ZdZMjLQSFLqDMXz2fBUk,4696
36
36
  moleditpy/modules/molecular_data.py,sha256=8gE9ByYg3kSBfb1zANsyad_BVBTm6WOLF7NsZIYuG2E,13250
37
- moleditpy/modules/molecule_scene.py,sha256=khdt7h9Mk_D1cMbYeHGtq7P9aFXo0xG-hcShU_H2Y-Q,95911
37
+ moleditpy/modules/molecule_scene.py,sha256=7NijkUjelz7ZrGxZxaPPGMrMFRB8kw_eLQ0gXqge9BY,96429
38
38
  moleditpy/modules/move_group_dialog.py,sha256=Fyuy3Uq1KsFsk9qR96r_FxPbAM_-zSfW2dsMQGv7btc,27276
39
39
  moleditpy/modules/periodic_table_dialog.py,sha256=ItEZUts1XCietz9paY-spvbzxh6SXak3GnikwqkHZCw,4006
40
40
  moleditpy/modules/planarize_dialog.py,sha256=eaqI1MpF35e-VUMpJATt-EtGG5FhcSUlbAenUaFGabY,8593
@@ -51,9 +51,9 @@ moleditpy/modules/assets/file_icon.ico,sha256=yyVj084A7HuMNbV073cE_Ag3Ne405qgOP3
51
51
  moleditpy/modules/assets/icon.icns,sha256=wD5R6-Vw7K662tVKhu2E1ImN0oUuyAP4youesEQsn9c,139863
52
52
  moleditpy/modules/assets/icon.ico,sha256=RfgFcx7-dHY_2STdsOQCQziY5SNhDr3gPnjO6jzEDPI,147975
53
53
  moleditpy/modules/assets/icon.png,sha256=kCFN1WacYIdy0GN6SFEbNA00ef39pCczBnFdkkBI8Bs,147110
54
- moleditpy-2.4.0.dist-info/licenses/LICENSE,sha256=OXLcl0T2SZ8Pmy2_dmlvKuetivmyPd5m1q-Gyd-zaYY,35149
55
- moleditpy-2.4.0.dist-info/METADATA,sha256=yzgQyEPi642KrEMUGtKGerD1OOw-BLlLVe044KCz5l0,60629
56
- moleditpy-2.4.0.dist-info/WHEEL,sha256=qELbo2s1Yzl39ZmrAibXA2jjPLUYfnVhUNTlyF1rq0Y,92
57
- moleditpy-2.4.0.dist-info/entry_points.txt,sha256=yH1h9JjALhok1foXT3-hYrC4ufoZt8b7oiBcsdnGNNM,54
58
- moleditpy-2.4.0.dist-info/top_level.txt,sha256=ARICrS4ihlPXqywlKl6o-oJa3Qz3gZRWu_VZsQ3_c44,10
59
- moleditpy-2.4.0.dist-info/RECORD,,
54
+ moleditpy-2.4.2.dist-info/licenses/LICENSE,sha256=OXLcl0T2SZ8Pmy2_dmlvKuetivmyPd5m1q-Gyd-zaYY,35149
55
+ moleditpy-2.4.2.dist-info/METADATA,sha256=5CnCyD09YUOEIhCrK9sIRZNsbP2iqsPP4K9RWwZezYM,60629
56
+ moleditpy-2.4.2.dist-info/WHEEL,sha256=qELbo2s1Yzl39ZmrAibXA2jjPLUYfnVhUNTlyF1rq0Y,92
57
+ moleditpy-2.4.2.dist-info/entry_points.txt,sha256=yH1h9JjALhok1foXT3-hYrC4ufoZt8b7oiBcsdnGNNM,54
58
+ moleditpy-2.4.2.dist-info/top_level.txt,sha256=ARICrS4ihlPXqywlKl6o-oJa3Qz3gZRWu_VZsQ3_c44,10
59
+ moleditpy-2.4.2.dist-info/RECORD,,