Форум программистов, компьютерный форум, киберфорум
C++ Builder
Войти
Регистрация
Восстановить пароль
Блоги Сообщество Поиск  
 
 
Рейтинг 5.00/18: Рейтинг темы: голосов - 18, средняя оценка - 5.00
28 / 24 / 10
Регистрация: 26.04.2015
Сообщений: 307

Дочерние формы (не MDI) и STL::List

22.08.2016, 23:53. Показов 3997. Ответов 63
Метки list (Все метки)

Студворк — интернет-сервис помощи студентам
Хай всем, нашёл тему по форуму, близкую к моей задаче:
Динамическое создание и удаление компонентов.,
но возник вопрос, корректно ли использование того кода для компонентов формы
C++
1
2
3
4
5
6
7
8
void __fastcall TfmSendFile::FormCreate(TObject *Sender)
{
    sbtDel = new TsButton(fmSendFile);
    sbtDel->OnClick = sbtDelClick;
 
    Number = 0;
    vLabel.push_back(new TsLabel(fmSendFile));
...
непосредственно к самой форме? (у меня в программе - список указателей дочерних окон главной формы). И ещё, как удалять из списка эти указатели?
Unit1.cpp
C++
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
//===========================================================================
#include <vcl.h>
#include <list.h>
#pragma hdrstop
#include "Unit1.h"
 
#pragma package(smart_init)
#pragma resource "*.dfm"
TForm1 *Form1;
    list<TForm1*> myList; 
 
__fastcall TForm1::TForm1(TComponent* Owner)
    : TForm(Owner)
{
    frm_2 = NULL; //(TForm*) - указатель на дочернюю форму
    ed = NULL;    //(TEdit*) - указатель на Edit этой формы
}
//===========================================================================
 
void __fastcall TForm1::Button1Click(TObject *Sender)
{
   
//    if ( frm_2 == NULL )
    {
    frm_2 = new TForm( this );            
//  frm_2->Parent = this;
    frm_2->Top = (ClientHeight - frm_2->Height)*0.50; 
    frm_2->Left = (ClientWidth - frm_2->Width)*0.50;
    
//  frm_2->ActiveControl = NULL;
    ::SetParent( frm_2->Handle, Handle ); 
    frm_2->OnClose = &Form2Close;         
    frm_2->Show();                       
 
        ed = new TEdit( frm_2 );           
    ed->Parent = frm_2;
        ed->Top = 20;                         
    ed->Left = 30;
    ed->Font->Style = TFontStyles() << fsBold;    
    }
//--------------------------
    myList.push_back(new TForm1(frm_2)); // <--- вот это интересует в сравнении с темой про компоненты
//--------------------------
}
//---------------------------------------------------------------------------
 
 
void __fastcall TForm1::Form2Close(TObject *Sender, TCloseAction &Action)
{
    if (ed!=NULL)
        delete ed;     
    ed = NULL;     
 
    if (frm_2!=NULL)   
        delete frm_2;
    frm_2 = NULL;     
}
//---------------------------------------------------------------------------
 
void __fastcall TForm1::WndProc(TMessage &Message)
{
    static bool isAppActive = true;
 
    switch(Message.Msg)
    {
        case WM_NCACTIVATE:
            Message.WParam = isAppActive;
            Message.Result = DefWindowProc(this->Handle, Message.Msg, Message.WParam, Message.LParam);
            break;
 
        case WM_ACTIVATEAPP:
            isAppActive = (Message.WParam != 0);
            LPARAM lp = 0L;
            this->Perform(WM_NCACTIVATE, 0, lp);
            Message.Result = 0;
            return;
    }
    TForm::WndProc(Message); // interited
}
//---------------------------------------------------------------------------
Unit1.h
C++
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
//===========================================================================
#ifndef     Unit1H
#define     Unit1H
 
#include <Classes.hpp>
#include <Controls.hpp>
#include <StdCtrls.hpp>
#include <Forms.hpp>
#include <ExtCtrls.hpp>
 
//---------------------------------------------------------------------------
class   TForm1 : public TForm
{
__published:    // IDE-managed Components
    TButton *Button1;
    TEdit *Edit1;
    void __fastcall Button1Click(TObject *Sender);
 
private:    // User declarations
    TForm* frm_2;    
    void __fastcall TForm1::Form2Close(TObject *Sender, TCloseAction &Action);  
    TEdit* ed;    
 
protected:
        void __fastcall WndProc(TMessage &Message);
 
public:     // User declarations
    __fastcall  TForm1(TComponent* Owner); 
};
//---------------------------------------------------------------------------
extern PACKAGE TForm1 *Form1;
//---------------------------------------------------------------------------
#endif  //(Unit1H)
//===========================================================================
0
cpp_developer
Эксперт
20123 / 5690 / 1417
Регистрация: 09.04.2010
Сообщений: 22,546
Блог
22.08.2016, 23:53
Ответы с готовыми решениями:

Работа с MDI и Дочерние формы
Здравствуйте у меня вопрос, есть 2 формы,главная и дочерняя. Дочерняя выводится через StripMenu. Когда я вывожу не одну дочернюю, а...

MDI вместо закрытия главной формы закрываются дочерние
Собственно есть главная &quot;MainForm&quot; и несколько дочерних, генерируемых динамически, у обоих типов форм есть обработчик закрытия. Проблема в...

Как сделать что бы дочерние формы в MDI application delphi сразу не открывались при запуске программы
Здравствуйте, как сделать что бы дочерние формы в MDI application delphi сразу не открывались при запуске программы, а открывались после...

63
Модератор
 Аватар для vxg
3410 / 2182 / 354
Регистрация: 13.01.2012
Сообщений: 8,468
24.08.2016, 12:13
Студворк — интернет-сервис помощи студентам
Fulcrum_013, он не будет обрабатывать удаление формы - мы просто через remove уберём из списка указатель, за удаление формы после закрытия отвечает она сама, вы читаете посты?
0
 Аватар для Fulcrum_013
2083 / 1575 / 169
Регистрация: 14.12.2014
Сообщений: 13,614
24.08.2016, 12:27
TList кстати непригоден по той же причине.

Добавлено через 2 минуты
Цитата Сообщение от vxg Посмотреть сообщение
он не будет обрабатывать удаление формы - мы просто через remove уберём из списка указатель,
Зачем изобретать велосипед если уже есть специально предназначенные для этих целей автоматические средства?

Добавлено через 9 минут
Если уж хочется отдельный список делать то делайте TObjectList или TComponentList которые отслеживают внешнее удаление содержимого.
0
Модератор
 Аватар для vxg
3410 / 2182 / 354
Регистрация: 13.01.2012
Сообщений: 8,468
24.08.2016, 12:29
Fulcrum_013, к сожалению я не могу сказать прямым текстом что за чувства вы будите во мне) глядите. У нас создаются формы. После закрытия они сами удаляются. Владелец нам в такой схеме не нужен. По какой-то причине нам нужен список тех форм что мы наплодили (зачем - хз, придумайте что нибудь, например для кнопки закрыть все порождённые окна). Что бы этот список был актуален мы при создании формы помещаем в него указатель, а при закрытии формы она удаляет указатель на саму себя из списка. Для этой цели подойдёт любой список. Более того, если охото можно все-таки назначать владельца и перебирать хранимый им список дочек. Что тут на ваш взгляд не так?
1
 Аватар для Fulcrum_013
2083 / 1575 / 169
Регистрация: 14.12.2014
Сообщений: 13,614
24.08.2016, 12:37
Цитата Сообщение от vxg Посмотреть сообщение
а при закрытии формы она удаляет указатель на саму себя из списка
А если пользуете в качестве списка в котором их содержите TComponentList то можете просто в обработчике закрытия написать Action=caFree;. Извлечется она из списка при этом сама, оповестив контейнер из деструктора. причем где бы ее не удалили. В этом и прелесть умных контейнеров и умных объектов - удалил и за висячие ссылки можешь не беспокоится в отличии от контейнеров std
0
Модератор
 Аватар для vxg
3410 / 2182 / 354
Регистрация: 13.01.2012
Сообщений: 8,468
24.08.2016, 12:38
Fulcrum_013, покажите код
0
Супер-модератор
Эксперт Pascal/DelphiАвтор FAQ
 Аватар для volvo
33450 / 21550 / 8248
Регистрация: 22.10.2011
Сообщений: 37,002
Записей в блоге: 12
24.08.2016, 12:41
Какой код? Описание смотрим: TComponentList
0
Модератор
 Аватар для vxg
3410 / 2182 / 354
Регистрация: 13.01.2012
Сообщений: 8,468
24.08.2016, 13:20
Цитата Сообщение от volvo Посмотреть сообщение
Какой код?
простой код. кодистый такой. вы предлагаете размещать в TComponentList указатели на формы и эти формы при вызове своих деструкторов сами удалят себя из списка?

Добавлено через 2 минуты
сами я имею ввиду волшебным образом - просто потому что список дружественным образом как то под капотом зарегистрировался на них. то есть удалят без нашего участия.
0
 Аватар для Fulcrum_013
2083 / 1575 / 169
Регистрация: 14.12.2014
Сообщений: 13,614
24.08.2016, 13:39
Цитата Сообщение от vxg Посмотреть сообщение
вы предлагаете размещать в TComponentList указатели на формы и эти формы при вызове своих деструкторов сами удалят себя из списка?
Да именно так. Это азы VCL.
примерно вот так
C++
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
class TForm1:public TForm{
        ...
private:
         TComponentList *OpenForms=nullptr;       
};
void TForm1::FormCreate(TObject* Sender){
       OpenForms=new TComponentList();
}
void TForm1::ButtonClick(TObject* Sender){
       TForm *Form=new TForm(nullptr);
        OpenForms->Add(Form) ;
       ....
}
void __fastcall TForm1::Form2Close(TObject *Sender, TCloseAction &Action){
      Action=caFree;
}
void __fastcall TForm1::CloseAllClick(TObject *Sender){
     OpenForms->Clear();
}
void __fastcall TForm1::FormDestroy(TObject *Sender){
      delete OpenForms;
}
Добавлено через 3 минуты
Цитата Сообщение от vxg Посмотреть сообщение
то есть удалят без нашего участия.
Да имено так.
0
Супер-модератор
Эксперт Pascal/DelphiАвтор FAQ
 Аватар для volvo
33450 / 21550 / 8248
Регистрация: 22.10.2011
Сообщений: 37,002
Записей в блоге: 12
24.08.2016, 13:40
Цитата Сообщение от vxg Посмотреть сообщение
вы предлагаете размещать в TComponentList указатели на формы и эти формы при вызове своих деструкторов сами удалят себя из списка?
Да, именно так и будет. Проверить негде, но не сомневаюсь, что код заработает:
C++
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
void __fastcall TForm1::MyOnClose(TObject *Sender, TCloseAction &Action)
{
    Action = caFree;
}
 
 
void __fastcall TForm1::btnCreateFormsClick(TObject *Sender)
{
    for(int i = 0; i < 10; i++)
    {
        TForm *frm = new TForm(this);
        frm->Show();
        frm->Caption = IntToStr(i);
        frm->OnClose = MyOnClose;
        lst->Add(frm); // TComponentList *lst, не забудь создать в конструкторе главной формы
    }
}
//---------------------------------------------------------------------------
 
 
void __fastcall TForm1::btnShowListClick(TObject *Sender)
{
    Memo1->Clear();
    for(int i = 0; i < lst->Count; i++)
    {
        Memo1->Lines->Add(static_cast<TForm*>(lst->Items[i])->Caption);
    }
}
Создай формы, посмотри содержимое списка, потом удали любую дочернюю, на выбор, и еще раз посмотри, что осталось в списке...
0
Модератор
 Аватар для vxg
3410 / 2182 / 354
Регистрация: 13.01.2012
Сообщений: 8,468
24.08.2016, 14:09
Fulcrum_013, volvo, да, TComponentList действительно смотрится органичнее в этом смысле, согласен. не умоляя красоты вашего решения скажу лишь что ходили мы вокруг да около ровно одной строки
C++
1
super_form->forms.remove(this)
добавляемой после
C++
1
Action = caFree
посему хочется вспомнить слова ковбоя из анекдота: "это мы получается бесплатно коровьих лепешек наелись ?"
0
 Аватар для Fulcrum_013
2083 / 1575 / 169
Регистрация: 14.12.2014
Сообщений: 13,614
24.08.2016, 14:21
Цитата Сообщение от vxg Посмотреть сообщение
не умоляя красоты вашего решения скажу лишь что ходили мы вокруг да около ровно одной строки
И кучи ненужных телодвижений на "закрыть все" "закрыть главную форму" и тд
0
24.08.2016, 14:24

Не по теме:

Fulcrum_013, вы хотели сказать и еще одной двух строчек при закрыть все... и из-за этих двух строчек торгуется человек который непонятно зачем написал аналог того же самого TComponentList)

0
 Аватар для Fulcrum_013
2083 / 1575 / 169
Регистрация: 14.12.2014
Сообщений: 13,614
24.08.2016, 14:43

Не по теме:

Цитата Сообщение от vxg Посмотреть сообщение
непонятно зачем написал аналог того же самого TComponentList
Затем что во первых в MSVC его нет а паскаль компилить майкрософту борланд запрещает. Во вторых мой вариант заточен под быструю вставку/удаление без соблюдения упорядоченности а не под навороченный функционал при этом хранимый объект реализует только необходмый для конкретной задачи минимум функционала оповещений а не универсально для любого количества списков как TObject или TComponent [/CPP]



Добавлено через 8 минут

Не по теме:

Ну и написан он потому как программируя в другой среде эти одну-две (десятка при некоторых делах) строчки писать по сто раз запарился программируя не под VCL

0
 Аватар для SmittWesson
236 / 196 / 21
Регистрация: 04.06.2014
Сообщений: 1,309
24.08.2016, 19:03
Цитата Сообщение от vxg Посмотреть сообщение
55-я строка Unit1.cpp
Спасибо увидел. Просто наспех смотрел, сразу не заметил.
0
28 / 24 / 10
Регистрация: 26.04.2015
Сообщений: 307
24.08.2016, 20:28  [ТС]
Цитата Сообщение от vxg Посмотреть сообщение
Кстати эдит убивать не нужно если вы в его конструктор подали форму на которой он лежит - его убьёт сама форма в своём деструкторе
простите не подумал об этом, просто привык все new подчищать delete'ами, спс за совет.
//-------------------------------------------
Цитата Сообщение от vxg Посмотреть сообщение
Fulcrum_013, caFree присваивать нужно безусловно и об этом сказано. caFree не имеет отношения к владельцу - форма должна удалится даже если владелец 0 - так зачем её обманывать и назначать владельцем того кто так никогда ничего с ней и не сделает? Контейнер (тут он один - список) нужен лишь для хранения указателей. То что мы храним в нем указатель на форму никак не влияет на саму форму. То что мы удаляем указатель ИЗ СПИСКА никак не влияет на форму. Явное удаление формы как сказано выше в данном случае нежелательно и возложено на обработчик закрытия формы при условии присвоения Action значения caFree. Так что же вам тут не нравится? То что вместо TList или вообще TForm[] использовали std::list? Это дело вкуса и к задаче отношения не имеет.
debug показывает значение в Action - caHide, я так понял по дефорту оно идёт? Добавил в обработчик закрытия строчку:
C++
1
2
3
4
void __fastcall TForm1::Form2Close(TObject *Sender, TCloseAction &Action)
{
Action = caFree;
...
//-------------------------------------------
Цитата Сообщение от Fulcrum_013 Посмотреть сообщение
Только STL для этого не нужна
Разве TList'у не желательно знать свои точные размеры? По дефолту у него Capacity=4 же? И при каждом добавлении нового объекта сверх этого числа будет перестроение в памяти. В моей задаче реальное к-во создаваемых объектов TForm зависит лишь от выбора пользователя - нужно 2 делаем 2, нужно 20 - создаём 20. Поэтому я и подумал, что STL лучше, единственное что выбор пал на List, а не vector, т.к. он поддерживает удаление с любой точки без сдвига элементов контейнера. Поправьте если я не прав.
//-------------------------------------------
Цитата Сообщение от Fulcrum_013 Посмотреть сообщение
То что он во первых не нужен а во вторых не сможет нормально обработать удаление формы без явного извлечения из него этой формы вручную
Этого не понял, поясните пож.
//-------------------------------------------
Цитата Сообщение от vxg Посмотреть сообщение
размещать в TComponentList указатели на формы и эти формы при вызове своих деструкторов сами удалят себя из списка?
Цитата Сообщение от vxg Посмотреть сообщение
Fulcrum_013, volvo, да, TComponentList действительно смотрится органичнее в этом смысле, согласен.
ни разу не использовал его... но если все сошлись на мнении, что он будет удобнее - то перефразирую свой код через него и отпишусь, что вышло.
За советы - всем гуру плюсов спс!
0
 Аватар для Fulcrum_013
2083 / 1575 / 169
Регистрация: 14.12.2014
Сообщений: 13,614
24.08.2016, 21:15
Цитата Сообщение от Wolver Посмотреть сообщение
Разве TList'у не желательно знать свои точные размеры?
Вопрос в том что Tlist не следит за жизненным циклом содержимого а TComponentList следит

Добавлено через 1 минуту
Цитата Сообщение от Wolver Посмотреть сообщение
так зачем её обманывать и назначать владельцем того кто так никогда ничего с ней и не сделает?
Для того чтобы она закрылась и удалилась когда удаляется или очищается список владения этого владельца. НАпример при закрытии формы которая ее открыла или по кнопке "закрыть все"

Добавлено через 2 минуты
Цитата Сообщение от Wolver Посмотреть сообщение
caFree не имеет отношения к владельцу
а сaFree действительно к владельцу отношения не имеет. Суть в чем - владелец удалит при своем удалении / очистке если ты не удалишь его другим способом. Если удалишь все поддерживающие слежение за жизненным циклом содержимого о ней тут же забудут и тебе по поводу висячих ссылок на нее заморачиваться не надо будет.
0
 Аватар для SmittWesson
236 / 196 / 21
Регистрация: 04.06.2014
Сообщений: 1,309
24.08.2016, 21:44
Цитата Сообщение от Wolver Посмотреть сообщение
простите не подумал об этом, просто привык все new подчищать delete'ами
В Билдере, при закрытии формы, делиты ставить совсем не нужно. Деструктор класса объекта всё сделает за вас. Достаточно прописать обработчик FormClose. Внутри него, убираем такие "штучки" типа static и тому подобные. К стати, статики уборщиком Винды не убираются. Как и функции с префиксом static. Лучше использовать динамические переменные и объекты из библиотек. Эти хоть, самоуничтожаются. Динамические, хоть и работают немного медленнее, но сейчас и годы не 1980-е .
0
279 / 156 / 52
Регистрация: 30.06.2011
Сообщений: 1,712
25.08.2016, 08:14
Цитата Сообщение от SmittWesson Посмотреть сообщение
статики уборщиком Винды не убираются
Даже в том случае, если приложение закроется?
0
Модератор
 Аватар для vxg
3410 / 2182 / 354
Регистрация: 13.01.2012
Сообщений: 8,468
25.08.2016, 08:20
d7d1cd, статика вообще не убирается, какой ещё уборщик? Статика это просто данные - инициализируются при запуске и умирают с программой
0
28 / 24 / 10
Регистрация: 26.04.2015
Сообщений: 307
25.08.2016, 21:21  [ТС]
Цитата Сообщение от volvo Посмотреть сообщение
Да, именно так и будет. Проверить негде, но не сомневаюсь, что код заработает:
Да, код прекрасно работает, единственное что они не дочерние т.к. упущено
C++
1
::SetParent(frm->Handle, Handle );
но не суть)
Возник вопрос по TComponentList, подскажите пож., при удалении элемента с Листа получается он автоматически сдвигает элементы и автоматически уменьшает Count? И т.е. если мне нужно будет проделать какие-то пользовательские действия с указателем на дочернее окно - их надо делать до его уничтожения?
Цитата Сообщение от vxg Посмотреть сообщение
То что мы храним в нем указатель на форму никак не влияет на саму форму. То что мы удаляем указатель ИЗ СПИСКА никак не влияет на форму.
В TComponentList получается наоборот?

Добавлено через 7 минут

Не по теме:

Цитата Сообщение от Fulcrum_013 Посмотреть сообщение
Да именно так. Это азы VCL.
примерно вот так
C++
1
2
3
...
TComponentList *OpenForms=nullptr;
...
привлёк внимание nullptr - оказывается довольно классная вещица из С++11 для указателей :) Было очень любопытно прочитать про неё, спс - http://archive.kalnitsky.org/2... 11-part-2/

0
Надоела реклама? Зарегистрируйтесь и она исчезнет полностью.
raxper
Эксперт
30234 / 6612 / 1498
Регистрация: 28.12.2010
Сообщений: 21,154
Блог
25.08.2016, 21:21

Дочерние окна MDI
На MDI форме есть кнопка, по нажатию на которую должны выстроиться каскадом все дочерние формы. метод Cascade – выстроить каскадом...

mdi - дочерние окна перетирают друг друга
Здравствуйте форумчане! Сталкнулся с небольшой проблемой для меня. А именно: Написал программу MDI,в каждом окне можно рисовать...

Как настроить соответствие размеров MDI-контейнера и MDI-формы?
Здравствуйте! Я для вкладки &quot;TbpDaSubekt&quot; сделал MDI-форму &quot;FrmTipHran&quot;: FrmTipHran.TopLevel = False FrmTipHran.Parent =...

Работа с STL list + пропись List.h
Здравствуйте! Помогите, пожалуйста, разобраться с STL list. Нужно сделать программу в которой будет содержаться класс Student, прописать...

Чтоб Дочерние формы открывались по иерархии от главной формы
Есть главная форма Form1, Form1 - topmost=true, т.е Form 1 отображается поверх всех окон. Нужно чтоб form2 открывалась не поверх всех окон...


Искать еще темы с ответами

Или воспользуйтесь поиском по форуму:
40
Ответ Создать тему
Новые блоги и статьи
Remote Connection Manager
DevAlt 21.06.2026
Написал для себя небольшую прилагу: https:/ / github. com/ altbodhi/ ReConMan По итогу пришел к мысли, что DU не дружат с существующими технологиями. От сериализации до отображения в реляционную. . .
Администрация Хабра удаляет новые энрегоэфективные алгоритмы, которые не западной школы кода, и вовсе никак не сгенерировавны.
Hrethgir 20.06.2026
Делается это, как замечено, при правках - при объявлении концептуальных отличий в алгоримах. Делается это, по линейке событий - после дополнения публикации основными отличиями от основных западных. . .
Процесс ориентированная диалектика (не новость - просто системное обновление, философия).
Hrethgir 20.06.2026
Однажды один участник в своём блоге, на этом форуме, сделал запись "О языках замолвите слово". Понимая, что язык - важная вещь, я решил хорошо подумать, прежде чем сказать, и сказал то, что вы видите. . .
Контроль уникальности строк в табличной части документа
Maks 18.06.2026
Алгоритм из решения ниже разработан на примере нетипового документа "ПланированиеСпецтехники" с табличной частью "НаличиеОборудования", разработанного в КА2. Задача: контроль уникальности строк в. . .
Клиент
Uhbif79 18.06.2026
Здесь простой клиент для работы с сервером.
Сервер
Uhbif79 18.06.2026
Выкладываю простейший сервер.
Дефенестрация
kumehtar 18.06.2026
Узнал интересное слово. Дефенестрация. Это когда ты выбрасываешь кого-либо или что-либо из окна. Возьму на вооружение)))
Дихотомия добра и зла
kumehtar 18.06.2026
Как Дзен-буддисты говорят о добре и зле: не нужно воевать против зла, нужно воевать против невежества. Тогда добро станет ествественным, и поэтому вечным. Но дело в том, что невежество всё время. . .
КиберФорум - форум программистов, компьютерный форум, программирование
Powered by vBulletin
Copyright ©2000 - 2026, CyberForum.ru