App-InvestSim
view release on metacpan or search on metacpan
lib/App/InvestSim/GUI.pm view on Meta::CPAN
}
# Some global variables for some widgets that are accessed by other methods.
my $modes_combobox;
my @core_display_values;
my $display_table;
my $rent_cap_entry;
# Some variables that are linked to widgets, that are set from elsewhere.
my ($total_rent_text, $rent_cap_text, $pinel_worth_text, $notary_fees_text, $total_invested_text);
# How to format the various modes of the drop-down menu.
my @modes_format;
# Add in the given frame in column 0 and 1 a label and an entry text box in the
# given row that is incremented ($row is a ref to a scalar).
sub add_input_entry {
my ($frame, $row, $key, $text, $format, $tooltip) = @_;
my (undef, $validate) = @{$values_config{$key}};
my $var_ref = \$values{$key};
$frame->new_ttk__label(-text => "${text} :")
->g_grid(-column => 0, -row => $$row, -sticky => "e", -padx => "0 2");
my $e = $frame->new_ttk__entry(-width => ENTRY_WIDTH);
$e->g_grid(-column => 1, -row => $$row++, -sticky => "we", -pady => 2);
setup_entry($e, $var_ref, $format, $validate);
if ($tooltip) {
local $Text::Wrap::columns = 50;
$e->g_tooltip__tooltip(Text::Wrap::fill('', '', $tooltip));
}
}
# Build the main window of the app.
sub build {
my ($res_dir) = @_;
Tkx::package_require("tooltip");
Tkx::option_add("*tearOff", 0); # Disable obsolete tear-off menus
# For how to find existing background style, see: https://tkdocs.com/tutorial/styles.html#insidestyle
Tkx::ttk__style('map', 'TEntry', -fieldbackground => ['invalid', '#ff0000']);
Tkx::ttk__style('map', 'TEntry', -fieldbackground => ['readonly', '#eeeeee']);
Tkx::ttk__style('configure', 'Invalid.TEntry', -foreground => '#ff0000');
# This style is used for the entry in the main table, to make it clearer that
# clicking on them has an effect.
Tkx::ttk__style('map', 'DataTable.TEntry', -foreground => ['selected', '#0000ff']);
my $root = Tkx::widget->new(".");
# We start by hiding the root window, we will show it at the end, when all the
# UI has been built (to avoid an ugly effect where the user sees each control
# being added quickly to the UI).
$root->g_wm_withdraw();
$root->g_wm_resizable(0, 1); # Disable resizing of the window x.
# FIXME: Will have to be adapted to work in the PAR package.
if (Tkx::expr('$tcl_platform(platform)') eq 'windows') {
$root->g_wm_iconbitmap(catfile($res_dir, 'icon.ico'));
} else {
# On Linux, we cant read .ico files directly.
my $icon = Tkx::widget->new(Tkx::image_create_photo());
$icon->read(catfile($res_dir, 'sources', 'icon_32.png'));
# We could pass several images of different sizes here.
$root->g_wm_iconphoto($icon);
}
# We're copying the font used by the TreeView style and adding an 'bold'
# option to it, it will be used by the 'total' line.
my $default_treeview_font = Tkx::ttk__style('lookup', 'TreeView', '-font');
my $treeview_total_font = Tkx::font('create');
Tkx::font('configure', $treeview_total_font, Tkx::SplitList(Tkx::font('configure', $default_treeview_font)));
Tkx::font('configure', $treeview_total_font, -weight => 'bold');
# Build the left bar with various parameters.
{
my $frame = $root->new_ttk__frame(-padding => 3);
$frame->g_grid(-column => 0, -row => 0, -rowspan => 3, -sticky => "we");
my $row = 0;
for my $c (['invested', "Valeur du bien", \&format_euro, "Prix d'achat du bien, hors frais de notaire."],
['tax_rate', "Taux d'imposition marginal", \&format_percent],
['base_rent', "Loyer brut initial", \&format_euro],
['rent_charges', "Charge et gestion locative", \&format_percent],
['rent_increase', "revalorisation loyer", \&format_percent],
['duration', "Durée d'investissement", \&format_year],
['notary_fees', "Frais de notaire", \&format_percent],
['loan_insurance', "Assurance décès du prêt", \&format_percent],
['other_rate', "Taux de placement autre", \&format_percent],
['surface', "Superficie (pondérée)", \&format_surface, "Surface totale habitable, additionnée, le cas échéant, de la moitié des surfaces annexes dans la limite de 8m² (utilisée seulement pour l'application des plafonds 'Pinel')."]) {
add_input_entry($frame, \$row, @$c);
}
}
# Build the top bar with the loan duration and rate values.
my @loan_duration_texts; # Re-used in the core data table.
{
my $frame = $root->new_ttk__frame(-padding => 3);
$frame->g_grid(-column => 1, -row => 0, -sticky => "nwes");
$frame->g_grid_columnconfigure(0, -weight => 1);
$frame->new_ttk__label(-text => "Durée d'emprunt (années)")
->g_grid(-column => 0, -row => 0, -sticky => "e");
$frame->new_ttk__label(-text => "Taux d'emprunt")
->g_grid(-column => 0, -row => 1, -sticky => "e");
my $loan_durations = $values{loan_durations};
my $validate_duration = $values_config{loan_durations}[1];
my $loan_rates = $values{loan_rates};
my $validate_rate = $values_config{loan_rates}[1];
for my $i (0..NUM_LOAN_DURATION-1) {
my $d = $frame->new_ttk__entry(-width => ENTRY_WIDTH);
$d->g_grid(-column => $i + 1, -row => 0, -sticky => "we");
setup_entry($d, \$loan_durations->[$i], \&format_year, $validate_duration, \$loan_duration_texts[$i]);
my $r = $frame->new_ttk__entry(-width => ENTRY_WIDTH);
$r->g_grid(-column => $i + 1, -row => 1, -sticky => "we");
setup_entry($r, \$loan_rates->[$i], \&format_percent, $validate_rate);
}
}
# Build the combo-box with the list of possible values, in its own frame.
my @modes;
$modes[MONTHLY_PAYMENT] = "Mensualité de l'emprunt (assurance comprise)";
$modes_format[MONTHLY_PAYMENT] = \&format_euro;
$modes[LOAN_COST] = "Cout total de l'emprunt (assurance comprise)";
$modes_format[LOAN_COST] = \&format_euro;
$modes[YEARLY_RENT_AFTER_LOAN] = "Revenus locatif net déduit des remboursement, par an";
$modes_format[YEARLY_RENT_AFTER_LOAN] = \&format_euro;
$modes[MEAN_BALANCE_LOAN_DURATION] = "Balance mensuelle moyenne de l'opération sur la durée du prêt";
$modes_format[MEAN_BALANCE_LOAN_DURATION] = \&format_euro;
$modes[MEAN_BALANCE_OVERALL] = "Balance mensuelle moyenne de l'opération sur la durée de simulation";
$modes_format[MEAN_BALANCE_OVERALL] = \&format_euro;
$modes[NET_GAIN] = "Gain Net de l'opération";
$modes_format[NET_GAIN] = \&format_euro;
lib/App/InvestSim/GUI.pm view on Meta::CPAN
}
}
# Build the right bar with some other input values and the output values not
# depending on the loan parameters.
{
my $frame = $root->new_ttk__frame(-padding => 3);
$frame->g_grid(-column => 2, -row => 0, -rowspan => 3, -sticky => "we");
my $row = 0;
for my $c (['rent_delay', "Delai de mise en location", \&format_year],
['loan_delay', "Durée de franchise de l'emprunt", \&format_year],
['application_fees', "Frais de dossier du prêt", \&format_euro],
['mortgage_fees', "Frais d'hypothèque", \&format_percent],
['social_tax', "CSG + CRDS + Solidarité", \&format_percent]) {
add_input_entry($frame, \$row, @$c);
}
# Just some empty white-space between the inputs and the output fields.
$frame->g_grid_rowconfigure($row++, -minsize => 10);
$frame->new_ttk__label(-text => "Revenus total du loyer (net) :")
->g_grid(-column => 0, -row => $row, -sticky => "e", -padx => "0 2");
$frame->new_ttk__entry(-width => ENTRY_WIDTH, -state => 'readonly', -textvariable => \$total_rent_text, -takefocus => 0)
->g_grid(-column => 1, -row => $row++, -sticky => "we", -pady => 2);
$frame->new_ttk__label(-text => "Plafond du loyer :")
->g_grid(-column => 0, -row => $row, -sticky => "e", -padx => "0 2");
($rent_cap_entry = $frame->new_ttk__entry(-width => ENTRY_WIDTH, -state => 'readonly', -textvariable => \$rent_cap_text, -takefocus => 0))
->g_grid(-column => 1, -row => $row++, -sticky => "we", -pady => 2);
$frame->new_ttk__label(-text => "Valeur déductible du bien :")
->g_grid(-column => 0, -row => $row, -sticky => "e", -padx => "0 2");
$frame->new_ttk__entry(-width => ENTRY_WIDTH, -state => 'readonly', -textvariable => \$pinel_worth_text, -takefocus => 0)
->g_grid(-column => 1, -row => $row++, -sticky => "we", -pady => 2);
$frame->new_ttk__label(-text => "Frais de notaire :")
->g_grid(-column => 0, -row => $row, -sticky => "e", -padx => "0 2");
$frame->new_ttk__entry(-width => ENTRY_WIDTH, -state => 'readonly', -textvariable => \$notary_fees_text, -takefocus => 0)
->g_grid(-column => 1, -row => $row++, -sticky => "we", -pady => 2);
$frame->new_ttk__label(-text => "Montant total investi :")
->g_grid(-column => 0, -row => $row, -sticky => "e", -padx => "0 2");
$frame->new_ttk__entry(-width => ENTRY_WIDTH, -state => 'readonly', -textvariable => \$total_invested_text, -takefocus => 0)
->g_grid(-column => 1, -row => $row++, -sticky => "we", -pady => 2);
}
# Build the bottom table.
$root->g_grid_columnconfigure(1, -weight => 1);
$root->g_grid_rowconfigure(3, -weight => 1);
{
my $frame = $root->new_ttk__frame(-padding => 3, -height => 550);
$frame->g_grid_propagate(0);
$frame->g_grid(-column => 0, -row => 3, -columnspan => 3, -sticky => "nwes");
$frame->g_grid_columnconfigure(0, -weight => 1);
$frame->g_grid_rowconfigure(0, -weight => 1);
$display_table = $frame->new_ttk__treeview(-height => ($values{duration} // 20) + 2);
$display_table->g_grid(-column => 0, -row => 0, -rowspan => 2, -sticky => "nwes");
# We're setting a specific font for items with the tag 'total'.
$display_table->m_tag('configure', 'total', -font => $treeview_total_font);
#my $hscroll = $frame->new_tk__scrollbar(-orient => "horizontal", -command => [$display_table, "xview"]);
#$hscroll->g_grid(-column => 0, -row => 1, -sticky => "we");
my $vscroll = $frame->new_ttk__scrollbar(-orient => "vertical", -command => [$display_table, "yview"]);
$vscroll->g_grid(-column => 1, -row => 0, -sticky => "ns");
$frame->new_ttk__sizegrip()->g_grid(-column => 1, -row => 1, -sticky => "se");
$display_table->configure(-yscrollcommand => [$vscroll, "set"]);
#$display_table->configure(-xscrollcommand => [$hscroll, "set"]);
my @headings = ('Année', 'Loyer net', 'Placements', 'Principal du prêt', 'Intérêts du prêt', 'Frais du prêt', 'Revenus imposable', 'Déficit déductible', 'Impôt', 'Solde annuel', 'Capital');
# We're not using the name of the columns (c1, c2, ...) we're only using their
# index (#0, #1, ...), including #0 the index of the first implicit column.
$display_table->m_configure(-columns => [map { "c$_" } 1..$#headings ]);
for my $c (0..$#headings) {
my $width = Tkx::font_measure(Tkx::ttk__style_lookup('Heading', '-font'), $headings[$c]);
$display_table->m_heading("#${c}", -text => $headings[$c]);
$display_table->m_column("#${c}", -width => $width, -anchor => 'e');
}
}
# Finally, we create a small menu.
{
my $menu = $root->new_menu;
$root->configure(-menu => $menu);
my $file = $menu->new_menu;
$menu->m_add_cascade(-menu => $file, -label => "Fichier", -underline => 0);
$file->m_add_command(-label => "Nouveau", -accelerator => 'Ctrl+N', -underline => 0,
-command => sub { init_values(); refresh_all_fields() });
$root->g_bind('<Control-n>', sub { init_values(); refresh_all_fields() });
$file->m_add_command(-label => "Ouvrir...", -accelerator => 'Ctrl+O', -underline => 0,
-command => sub { open_values(); refresh_all_fields() });
$root->g_bind('<Control-o>', sub { open_values(); refresh_all_fields() });
$file->m_add_command(-label => "Enregistrer", -accelerator => 'Ctrl+S', -underline => 0, -command => \&save_values);
$root->g_bind('<Control-s>', \&save_values);
$file->m_add_command(-label => "Enregistrer sous...", -accelerator => 'Ctrl+Alt+S', -underline => 12,-command => \&save_values_as);
$root->g_bind('<Control-Alt-s>', \&save_values_as);
$file->add_separator();
$file->m_add_command(-label => "Quitter", -accelerator => 'Alt+F4', -underline => 0,-command => sub { $root->g_destroy() });
# The binding for Alt-F4 is automatically supplied by Windows and can't be
# overriden. It will destroy the window. We catch it as well as the menu entry
# using the following bind command.
# If we bind to $root, then the event triggers for all contained widget.
$root->g_bind('<Destroy>', [sub { autosave() if $_[0] eq '.' }, Tkx::Ev('%W')]);
my $options = $menu->new_menu;
$menu->m_add_cascade(-menu => $options, -label => "Options", -underline => 0);
my $automatic_duration = \$values{automatic_duration};
$options->m_add_checkbutton(-label => "Durée automatique", -variable => $automatic_duration, -onvalue => 1, -offvalue => 0, -accelerator => 'Ctrl+D');
$root->g_bind('<Control-d>', sub { $$automatic_duration = 1 - $$automatic_duration });
my $taxes = $menu->new_menu;
$menu->m_add_cascade(-menu => $taxes, -label => "Fiscalité", -underline => 1);
my $pinel_menu = $taxes->new_menu;
my @pinel_zone = ('Zone A bis', 'Zone A', 'Zone B1', 'Zone B2');
my $disable_pinel_zone = sub {
for my $z (@pinel_zone) {
$pinel_menu->m_entryconfigure($z, -state => 'disabled');
}
};
my $enable_pinel_zone = sub {
for my $z (@pinel_zone) {
$pinel_menu->m_entryconfigure($z, -state => 'normal');
}
};
$taxes->m_add_cascade(-menu => $pinel_menu, -label => "Loi Pinel", -underline => 4);
my $pinel_duration = \$values{pinel_duration};
# TODO: test if loading a file with pinel duration 0 results in having the zone disabled correctly.
$pinel_menu->add_radiobutton(-label => "Non", -variable => $pinel_duration, -value => 0, -command => sub { $disable_pinel_zone->(); calculate_all() });
$pinel_menu->add_radiobutton(-label => "6 ans", -variable => $pinel_duration, -value => 6, -command => sub { $enable_pinel_zone->(); calculate_all() });
$pinel_menu->add_radiobutton(-label => "9 ans", -variable => $pinel_duration, -value => 9, -command => sub { $enable_pinel_zone->(); calculate_all() });
$pinel_menu->add_radiobutton(-label => "12 ans", -variable => $pinel_duration, -value => 12, -command => sub { $enable_pinel_zone->(); calculate_all() });
$pinel_menu->add_separator();
my $pinel_zone = \$values{pinel_zone};
for my $i (0..$#pinel_zone) {
( run in 1.649 second using v1.01-cache-2.11-cpan-ceb78f64989 )