e9bc3df09b
svn: r13707
652 lines
24 KiB
Python
652 lines
24 KiB
Python
#
|
|
# Gramps - a GTK+/GNOME based genealogy program
|
|
#
|
|
# Copyright (C) 2004-2007 Donald N. Allingham
|
|
# Copyright (C) 2008 Brian G. Matherly
|
|
# Contribution 2009 by Brad Crittenden <brad [AT] bradcrittenden.net>
|
|
#
|
|
# This program is free software; you can redistribute it and/or modify
|
|
# it under the terms of the GNU General Public License as published by
|
|
# the Free Software Foundation; either version 2 of the License, or
|
|
# (at your option) any later version.
|
|
#
|
|
# This program is distributed in the hope that it will be useful,
|
|
# but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
# GNU General Public License for more details.
|
|
#
|
|
# You should have received a copy of the GNU General Public License
|
|
# along with this program; if not, write to the Free Software
|
|
# Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
|
|
#
|
|
|
|
# Written by B.Malengier
|
|
|
|
#-------------------------------------------------------------------------
|
|
#
|
|
# Python modules
|
|
#
|
|
#-------------------------------------------------------------------------
|
|
import os
|
|
import sys
|
|
from gettext import gettext as _
|
|
|
|
#-------------------------------------------------------------------------
|
|
#
|
|
# set up logging
|
|
#
|
|
#-------------------------------------------------------------------------
|
|
import logging
|
|
log = logging.getLogger(".ExportAssistant")
|
|
|
|
#-------------------------------------------------------------------------
|
|
#
|
|
# Gnome modules
|
|
#
|
|
#-------------------------------------------------------------------------
|
|
import gtk
|
|
#-------------------------------------------------------------------------
|
|
#
|
|
# gramps modules
|
|
#
|
|
#-------------------------------------------------------------------------
|
|
|
|
import const
|
|
import config
|
|
from gui.pluginmanager import GuiPluginManager
|
|
import Utils
|
|
import ManagedWindow
|
|
|
|
#-------------------------------------------------------------------------
|
|
#
|
|
# Constants
|
|
#
|
|
#-------------------------------------------------------------------------
|
|
_gramps_png = os.path.join(const.IMAGE_DIR,"gramps.png")
|
|
_splash_jpg = os.path.join(const.IMAGE_DIR,"splash.jpg")
|
|
|
|
#-------------------------------------------------------------------------
|
|
#
|
|
# ExportAssistant
|
|
#
|
|
#-------------------------------------------------------------------------
|
|
|
|
_ExportAssistant_pages = {
|
|
'intro' : 0,
|
|
'exporttypes' : 1,
|
|
'options' : 2,
|
|
'fileselect' : 3,
|
|
'confirm' : 4,
|
|
'summary' : 5,
|
|
}
|
|
|
|
class ExportAssistant(gtk.Assistant, ManagedWindow.ManagedWindow) :
|
|
"""
|
|
This class creates a GTK assistant to guide the user through the various
|
|
Save as/Export options.
|
|
|
|
The overall goal is to keep things simple by presenting few choice options
|
|
on each assistant page.
|
|
|
|
The export formats and options are obtained from the plugins.
|
|
|
|
"""
|
|
|
|
__gsignals__ = {"apply": "override", "cancel": "override",
|
|
"close": "override", "prepare": "override"}
|
|
|
|
def __init__(self,dbstate,uistate):
|
|
"""
|
|
Set up the assistant, and build all the possible assistant pages.
|
|
|
|
Some page elements are left empty, since their contents depends
|
|
on the user choices and on the success of the attempted save.
|
|
|
|
"""
|
|
self.dbstate = dbstate
|
|
self.uistate = uistate
|
|
|
|
self.writestarted = False
|
|
|
|
#set up Assistant
|
|
gtk.Assistant.__init__(self)
|
|
##workaround around bug http://bugzilla.gnome.org/show_bug.cgi?id=56070
|
|
self.forward_button = None
|
|
gtk.Assistant.forall(self, self.get_forward_button)
|
|
## end
|
|
|
|
#set up ManagedWindow
|
|
self.top_title = _("Export Assistant")
|
|
ManagedWindow.ManagedWindow.__init__(self,uistate,[],
|
|
self.__class__)
|
|
self.set_window(self, None, self.top_title, isWindow=True)
|
|
|
|
#set up callback method for the export plugins
|
|
self.callback = self.pulse_progressbar
|
|
|
|
if self.dbstate.active:
|
|
self.person = self.dbstate.get_active_person()
|
|
else:
|
|
self.person = self.dbstate.db.find_initial_person()
|
|
|
|
self.logo = gtk.gdk.pixbuf_new_from_file(_gramps_png)
|
|
self.splash = gtk.gdk.pixbuf_new_from_file(_splash_jpg)
|
|
|
|
pmgr = GuiPluginManager.get_instance()
|
|
self.__exporters = pmgr.get_export_plugins()
|
|
self.map_exporters = {}
|
|
|
|
self.__previous_page = -1
|
|
|
|
#create the assistant pages
|
|
self.create_page_intro()
|
|
self.create_page_exporttypes()
|
|
self.create_page_options()
|
|
self.create_page_fileselect()
|
|
self.create_page_confirm()
|
|
#no progress page, looks ugly, and user needs to hit forward at end!
|
|
self.create_page_summary()
|
|
|
|
self.option_box_instance = None
|
|
#we need our own forward function as options page must not always be shown
|
|
self.set_forward_page_func(self.forward_func, None)
|
|
|
|
#ManagedWindow show method
|
|
ManagedWindow.ManagedWindow.show(self)
|
|
|
|
def get_forward_button(self, arg):
|
|
if isinstance(arg, gtk.HBox):
|
|
arg.forall(self._forward_btn)
|
|
|
|
def _forward_btn(self, arg):
|
|
if isinstance(arg, gtk.Button) and arg.get_label() == 'gtk-go-forward':
|
|
self.forward_button = arg
|
|
|
|
def build_menu_names(self, obj):
|
|
"""Override ManagedWindow method."""
|
|
return (self.top_title, None)
|
|
|
|
def create_page_intro(self):
|
|
"""Create the introduction page."""
|
|
label = gtk.Label(self.get_intro_text())
|
|
label.set_line_wrap(True)
|
|
label.set_use_markup(True)
|
|
|
|
page = label
|
|
page.show_all()
|
|
|
|
self.append_page(page)
|
|
self.set_page_header_image(page, self.logo)
|
|
self.set_page_side_image(page, self.splash)
|
|
self.set_page_title(page, _('Saving your data'))
|
|
self.set_page_complete(page, True)
|
|
self.set_page_type(page, gtk.ASSISTANT_PAGE_INTRO)
|
|
|
|
def create_page_exporttypes(self):
|
|
"""Create the export type page.
|
|
|
|
A Title label.
|
|
A table of format radio buttons and their descriptions.
|
|
|
|
"""
|
|
self.format_buttons = []
|
|
|
|
box = gtk.VBox()
|
|
box.set_border_width(12)
|
|
box.set_spacing(12)
|
|
|
|
table = gtk.Table(2*len(self.__exporters),2)
|
|
table.set_row_spacings(6)
|
|
table.set_col_spacings(6)
|
|
|
|
group = None
|
|
recent_type = config.get('behavior.recent-export-type')
|
|
|
|
exporters = [(x.get_name().replace("_", ""), x) for x in self.__exporters]
|
|
exporters.sort()
|
|
ix = 0
|
|
for sort_title, exporter in exporters:
|
|
title = exporter.get_name()
|
|
description= exporter.get_description()
|
|
self.map_exporters[ix] = exporter
|
|
button = gtk.RadioButton(group,title)
|
|
button.set_tooltip_text(description)
|
|
if not group:
|
|
group = button
|
|
self.format_buttons.append(button)
|
|
table.attach(button, 0, 2, 2*ix, 2*ix+1)
|
|
if ix == recent_type :
|
|
button.set_active(True)
|
|
ix += 1
|
|
|
|
box.add(table)
|
|
|
|
page = box
|
|
|
|
page.show_all()
|
|
|
|
self.append_page(page)
|
|
self.set_page_header_image(page, self.logo)
|
|
self.set_page_title(page, _('Choose the output format'))
|
|
|
|
self.set_page_type(page, gtk.ASSISTANT_PAGE_CONTENT)
|
|
|
|
|
|
def create_page_options(self):
|
|
# as we do not know yet what to show, we create an empty page
|
|
page = gtk.VBox()
|
|
page.set_border_width(12)
|
|
page.set_spacing(12)
|
|
|
|
page.show_all()
|
|
|
|
self.append_page(page)
|
|
self.set_page_header_image(page, self.logo)
|
|
self.set_page_complete(page, False)
|
|
self.set_page_type(page, gtk.ASSISTANT_PAGE_CONTENT)
|
|
|
|
def forward_func(self, pagenumber, data):
|
|
"""This function is called on forward press.
|
|
|
|
Normally, go to next page, however, before options,
|
|
we decide if options to show
|
|
"""
|
|
if pagenumber == _ExportAssistant_pages['exporttypes'] :
|
|
#decide if options need to be shown:
|
|
self.option_box_instance = None
|
|
ix = self.get_selected_format_index()
|
|
if not self.map_exporters[ix].get_config():
|
|
# no options needed
|
|
return pagenumber + 2
|
|
elif pagenumber == _ExportAssistant_pages['options']:
|
|
# need to check to see if we should show file selection
|
|
if (self.option_box_instance and
|
|
hasattr(self.option_box_instance, "no_fileselect")):
|
|
# don't show fileselect, but mark it ok
|
|
return pagenumber + 2
|
|
return pagenumber + 1
|
|
|
|
def create_options(self):
|
|
"""This method gets the option page, and fills it with the options."""
|
|
option = self.get_selected_format_index()
|
|
vbox = self.get_nth_page(_ExportAssistant_pages['options'])
|
|
(config_title, config_box_class) = self.map_exporters[option].get_config()
|
|
self.set_page_title(vbox, config_title)
|
|
# remove present content of the vbox
|
|
vbox.foreach(vbox.remove)
|
|
# add new content
|
|
if config_box_class:
|
|
self.option_box_instance = config_box_class(self.person)
|
|
box = self.option_box_instance.get_option_box()
|
|
vbox.add(box)
|
|
else:
|
|
self.option_box_instance = None
|
|
vbox.show_all()
|
|
|
|
# We silently assume all options lead to accepted behavior
|
|
self.set_page_complete(vbox, True)
|
|
|
|
def create_page_fileselect(self):
|
|
self.chooser = gtk.FileChooserWidget(gtk.FILE_CHOOSER_ACTION_SAVE)
|
|
#add border
|
|
self.chooser.set_border_width(12)
|
|
#global files, ask before overwrite
|
|
self.chooser.set_local_only(False)
|
|
self.chooser.set_do_overwrite_confirmation(True)
|
|
|
|
#created, folder and name not set
|
|
self.folder_is_set = False
|
|
|
|
#connect changes in filechooser with check to mark page complete
|
|
self.chooser.connect("selection-changed", self.check_fileselect)
|
|
self.chooser.connect("key-release-event", self.check_fileselect)
|
|
#first selection does not give a selection-changed event, grab the button
|
|
self.chooser.connect("button-release-event", self.check_fileselect)
|
|
#Note, we can induce an exotic error, delete filename,
|
|
# do not release button, click forward. We expect user not to do this
|
|
# In case he does, recheck on confirmation page!
|
|
|
|
self.chooser.show_all()
|
|
page = self.chooser
|
|
|
|
self.append_page(page)
|
|
self.set_page_header_image(page, self.logo)
|
|
self.set_page_title(page, _('Select Save File'))
|
|
#see if page can be set as complete :
|
|
self.check_fileselect(page)
|
|
self.set_page_type(page, gtk.ASSISTANT_PAGE_CONTENT)
|
|
|
|
def check_fileselect(self, filechooser, event=None, show=True):
|
|
"""Given a filechooser, determine if it can be marked complete in
|
|
the Assistant.
|
|
|
|
Used as normal callback and event callback. For callback, we will have
|
|
show=True
|
|
"""
|
|
filename = filechooser.get_filename()
|
|
folder = filechooser.get_current_folder()
|
|
#the file must be valid, not a folder, and folder must be valid
|
|
if filename and filename.strip and Utils.find_folder(filename) == '' \
|
|
and folder and Utils.find_folder(folder):
|
|
#this page of the assistant is complete
|
|
self.set_page_complete(filechooser, True)
|
|
##workaround around bug http://bugzilla.gnome.org/show_bug.cgi?id=56070
|
|
if self.forward_button and show:
|
|
self.forward_button.hide()
|
|
self.forward_button.show()
|
|
## end
|
|
|
|
else :
|
|
self.set_page_complete(filechooser, False)
|
|
|
|
def create_page_confirm(self):
|
|
# Construct confirm page
|
|
label = gtk.Label()
|
|
label.set_line_wrap(True)
|
|
label.set_use_markup(True)
|
|
label.show()
|
|
|
|
page = label
|
|
self.append_page(page)
|
|
self.set_page_header_image(page, self.logo)
|
|
self.set_page_title(page, _('Final confirmation'))
|
|
self.set_page_type(page, gtk.ASSISTANT_PAGE_CONFIRM)
|
|
self.set_page_complete(page, True)
|
|
|
|
def create_page_summary(self):
|
|
# Construct summary page
|
|
# As this is the last page needs to be of page_type
|
|
# gtk.ASSISTANT_PAGE_CONFIRM or gtk.ASSISTANT_PAGE_SUMMARY
|
|
page = gtk.Alignment(xalign=0.5, yalign=0.5, xscale=0,
|
|
yscale=0)
|
|
vbox = gtk.VBox()
|
|
vbox.set_border_width(12)
|
|
vbox.set_spacing(6)
|
|
self.labelsum = gtk.Label(_("Please wait while your data is selected and exported"))
|
|
self.labelsum.set_line_wrap(True)
|
|
self.labelsum.set_use_markup(True)
|
|
vbox.pack_start(self.labelsum, expand=True)
|
|
|
|
self.progressbar = gtk.ProgressBar()
|
|
vbox.pack_start(self.progressbar, expand=True)
|
|
|
|
page.add(vbox)
|
|
page.show_all()
|
|
|
|
self.append_page(page)
|
|
self.set_page_header_image(page, self.logo)
|
|
self.set_page_title(page, _('Summary'))
|
|
self.set_page_side_image(page, self.splash)
|
|
self.set_page_complete(page, False)
|
|
self.set_page_type(page, gtk.ASSISTANT_PAGE_SUMMARY)
|
|
|
|
|
|
def do_apply(self):
|
|
pass
|
|
|
|
def do_cancel(self):
|
|
if self.writestarted :
|
|
return True
|
|
else :
|
|
self.close()
|
|
|
|
def do_close(self):
|
|
if self.writestarted :
|
|
return True
|
|
else :
|
|
self.close()
|
|
|
|
def do_prepare(self, page):
|
|
"""
|
|
The "prepare" signal is emitted when a new page is set as the
|
|
assistant's current page, but before making the new page visible.
|
|
|
|
:param page: the new page to prepare for display.
|
|
|
|
"""
|
|
#determine if we go backward or forward
|
|
page_number = self.get_current_page()
|
|
assert page == self.get_nth_page(page_number)
|
|
if page_number <= self.__previous_page :
|
|
back = True
|
|
else :
|
|
back = False
|
|
|
|
if back :
|
|
#when moving backward, show page as it was,
|
|
#page we come from is set incomplete so as to disallow user jumping
|
|
# to last page after backward move
|
|
self.set_page_complete(self.get_nth_page(self.__previous_page),
|
|
False)
|
|
|
|
elif page_number == _ExportAssistant_pages['options'] :
|
|
self.create_options()
|
|
self.set_page_complete(page, True)
|
|
##workaround around bug http://bugzilla.gnome.org/show_bug.cgi?id=56070
|
|
if self.forward_button:
|
|
self.forward_button.hide()
|
|
self.forward_button.show()
|
|
## end
|
|
elif page == self.chooser :
|
|
# next page is the file chooser, reset filename, keep folder where user was
|
|
folder, name = self.suggest_filename()
|
|
if self.folder_is_set :
|
|
page.set_current_name(name)
|
|
else :
|
|
page.set_current_name(name)
|
|
page.set_current_folder(folder)
|
|
self.folder_is_set = True
|
|
# see if page is complete with above
|
|
self.check_fileselect(page, show=True)
|
|
|
|
elif self.get_page_type(page) == gtk.ASSISTANT_PAGE_CONFIRM :
|
|
# The confirm page with apply button
|
|
# Present user with what will happen
|
|
ix = self.get_selected_format_index()
|
|
format = self.map_exporters[ix].get_name()
|
|
page_complete = False
|
|
# If no file select:
|
|
if (self.option_box_instance and
|
|
hasattr(self.option_box_instance, "no_fileselect")):
|
|
# No file selection
|
|
filename = ''
|
|
confirm_text = _(
|
|
'The data will be exported as follows:\n\n'
|
|
'Format:\t%s\n\n'
|
|
'Press Apply to proceed, Back to revisit '
|
|
'your options, or Cancel to abort') % (format.replace("_",""), )
|
|
page_complete = True
|
|
else:
|
|
#Allow for exotic error: file is still not correct
|
|
self.check_fileselect(self.chooser, show=False)
|
|
if self.get_page_complete(self.chooser) :
|
|
filename = Utils.get_unicode_path(self.chooser.get_filename())
|
|
name = os.path.split(filename)[1]
|
|
folder = os.path.split(filename)[0]
|
|
confirm_text = _(
|
|
'The data will be saved as follows:\n\n'
|
|
'Format:\t%s\nName:\t%s\nFolder:\t%s\n\n'
|
|
'Press Apply to proceed, Back to revisit '
|
|
'your options, or Cancel to abort') % (format.replace("_",""), name, folder)
|
|
page_complete = True
|
|
else :
|
|
confirm_text = _(
|
|
'The selected file and folder to save to '
|
|
'cannot be created or found.\n\n'
|
|
'Press Back to return and select a valid filename.'
|
|
)
|
|
page_complete = False
|
|
# Set the page_complete status
|
|
self.set_page_complete(page, page_complete)
|
|
# If it is ok, then look for alternate confirm_text
|
|
if (page_complete and
|
|
self.option_box_instance and
|
|
hasattr(self.option_box_instance, "confirm_text")):
|
|
# Override message
|
|
confirm_text = self.option_box_instance.confirm_text
|
|
page.set_label(confirm_text)
|
|
|
|
elif self.get_page_type(page) == gtk.ASSISTANT_PAGE_SUMMARY :
|
|
# The summary page
|
|
# Lock page, show progress bar
|
|
self.pre_save(page)
|
|
# save
|
|
success = self.save()
|
|
# Unlock page
|
|
self.post_save()
|
|
|
|
#update the label and title
|
|
if success:
|
|
conclusion_title = _('Your data has been saved')
|
|
conclusion_text = _(
|
|
'The copy of your data has been '
|
|
'successfully saved. You may press Close button '
|
|
'now to continue.\n\n'
|
|
'Note: the database currently opened in your Gramps '
|
|
'window is NOT the file you have just saved. '
|
|
'Future editing of the currently opened database will '
|
|
'not alter the copy you have just made. ')
|
|
#add test, what is dir
|
|
conclusion_text += '\n\n' + _('Filename: %s') %self.chooser.get_filename()
|
|
else:
|
|
conclusion_title = _('Saving failed')
|
|
conclusion_text = _(
|
|
'There was an error while saving your data. '
|
|
'You may try starting the export again.\n\n'
|
|
'Note: your currently opened database is safe. '
|
|
'It was only '
|
|
'a copy of your data that failed to save.')
|
|
self.labelsum.set_label(conclusion_text)
|
|
self.set_page_title(page, conclusion_title)
|
|
self.set_page_complete(page, True)
|
|
else :
|
|
#whatever other page, if we show it, it is complete to
|
|
self.set_page_complete(page, True)
|
|
if page_number == _ExportAssistant_pages['exporttypes'] :
|
|
##workaround around bug http://bugzilla.gnome.org/show_bug.cgi?id=56070
|
|
if self.forward_button:
|
|
self.forward_button.hide()
|
|
self.forward_button.show()
|
|
## end
|
|
|
|
#remember previous page for next time
|
|
self.__previous_page = page_number
|
|
|
|
def close(self, *obj) :
|
|
#clean up ManagedWindow menu, then destroy window, bring forward parent
|
|
|
|
ManagedWindow.ManagedWindow.close(self,*obj)
|
|
|
|
def get_intro_text(self):
|
|
return _('Under normal circumstances, Gramps does not require you '
|
|
'to directly save your changes. All changes you make are '
|
|
'immediately saved to the database.\n\n'
|
|
'This process will help you save a copy of your data '
|
|
'in any of the several formats supported by Gramps. '
|
|
'This can be used to make a copy of your data, backup '
|
|
'your data, or convert it to a format that will allow '
|
|
'you to transfer it to a different program.\n\n'
|
|
'If you change your mind during this process, you '
|
|
'can safely press the Cancel button at any time and your '
|
|
'present database will still be intact.')
|
|
|
|
def get_selected_format_index(self):
|
|
"""
|
|
Query the format radiobuttons and return the index number of the
|
|
selected one.
|
|
|
|
"""
|
|
for ix in range(len(self.format_buttons)):
|
|
button = self.format_buttons[ix]
|
|
if button.get_active():
|
|
return ix
|
|
else:
|
|
return 0
|
|
|
|
def suggest_filename(self):
|
|
"""Prepare suggested filename and set it in the file chooser."""
|
|
ix = self.get_selected_format_index()
|
|
ext = self.map_exporters[ix].get_extension()
|
|
|
|
# Suggested folder: try last export, then last import, then home.
|
|
default_dir = config.get('paths.recent-export-dir')
|
|
if len(default_dir)<=1:
|
|
default_dir = config.get('paths.recent-import-dir')
|
|
if len(default_dir)<=1:
|
|
default_dir = const.USER_HOME
|
|
|
|
if ext == 'gramps':
|
|
new_filename = os.path.join(default_dir,'data.gramps')
|
|
elif ext == 'burn':
|
|
new_filename = os.path.basename(self.dbstate.db.get_save_path())
|
|
else:
|
|
new_filename = Utils.get_new_filename(ext,default_dir)
|
|
return (default_dir, os.path.split(new_filename)[1])
|
|
|
|
def save(self):
|
|
"""
|
|
Perform the actual Save As/Export operation.
|
|
|
|
Depending on the success status, set the text for the final page.
|
|
|
|
"""
|
|
if (self.option_box_instance and
|
|
hasattr(self.option_box_instance, "no_fileselect")):
|
|
filename = ""
|
|
else:
|
|
filename = Utils.get_unicode_path(self.chooser.get_filename())
|
|
config.set('paths.recent-export-dir', os.path.split(filename)[0])
|
|
ix = self.get_selected_format_index()
|
|
config.set('behavior.recent-export-type', ix)
|
|
export_function = self.map_exporters[ix].get_export_function()
|
|
success = export_function(self.dbstate.db,
|
|
filename,
|
|
self.option_box_instance,
|
|
self.callback)
|
|
return success
|
|
|
|
def pre_save(self,page):
|
|
#as all is locked, show the page, which assistant normally only does
|
|
# after prepare signal!
|
|
self.writestarted = True
|
|
page.set_child_visible(True)
|
|
self.show_all()
|
|
|
|
self.uistate.set_busy_cursor(1)
|
|
self.set_busy_cursor(1)
|
|
|
|
def post_save(self):
|
|
self.uistate.set_busy_cursor(0)
|
|
self.set_busy_cursor(0)
|
|
self.progressbar.hide()
|
|
self.writestarted = False
|
|
|
|
def set_busy_cursor(self,value):
|
|
"""Set or unset the busy cursor while saving data.
|
|
|
|
Note : self.window is the gtk.Assistant gtk.Window, not
|
|
a part of ManagedWindow
|
|
|
|
"""
|
|
if value:
|
|
self.window.set_cursor(gtk.gdk.Cursor(gtk.gdk.WATCH))
|
|
#self.set_sensitive(0)
|
|
else:
|
|
self.window.set_cursor(None)
|
|
#self.set_sensitive(1)
|
|
|
|
while gtk.events_pending():
|
|
gtk.main_iteration()
|
|
|
|
def pulse_progressbar(self, value, text=None):
|
|
self.progressbar.set_fraction(min(value/100.0, 1.0))
|
|
if text:
|
|
self.progressbar.set_text("%s: %d%%" % (text, value))
|
|
else:
|
|
self.progressbar.set_text("%d%%" % value)
|
|
while gtk.events_pending():
|
|
gtk.main_iteration()
|
|
|
|
|
|
|