diff --git a/.gitattributes b/.gitattributes index 85064b4e54..7c6a59ee91 100644 --- a/.gitattributes +++ b/.gitattributes @@ -6004,6 +6004,7 @@ docs/xml/lazutils/extendedstrings.xml svneol=native#text/plain docs/xml/lazutils/fileutil.xml svneol=native#text/plain docs/xml/lazutils/fpcadds.xml svneol=LF#text/xml eol=lf docs/xml/lazutils/fpmake.xml svneol=native#text/plain +docs/xml/lazutils/html2textrender.xml svneol=native#text/plain docs/xml/lazutils/laz2_dom.xml svneol=native#text/plain docs/xml/lazutils/laz2_xmlcfg.xml svneol=native#text/plain docs/xml/lazutils/laz2_xmlread.xml svneol=native#text/plain @@ -6064,6 +6065,7 @@ docs/xml/lazutils/ttprofile.xml svneol=native#text/plain docs/xml/lazutils/ttraster.xml svneol=native#text/plain docs/xml/lazutils/tttables.xml svneol=native#text/plain docs/xml/lazutils/tttypes.xml svneol=native#text/plain +docs/xml/lazutils/uitypes.xml svneol=native#text/plain docs/xml/lazutils/utf8process.xml svneol=native#text/plain docs/xml/lcl/actnlist.xml svneol=LF#text/xml eol=lf docs/xml/lcl/alllclunits.xml svneol=LF#text/xml eol=lf diff --git a/docs/xml/lazutils/extendedstrings.xml b/docs/xml/lazutils/extendedstrings.xml index 756b4f9b91..87a035b8c2 100644 --- a/docs/xml/lazutils/extendedstrings.xml +++ b/docs/xml/lazutils/extendedstrings.xml @@ -2,70 +2,78 @@ + ==================================================================== + ExtendedStrings + ==================================================================== + --> - Defines TExtendedStrings which is a normal TStringList, except that the Objects can hold any type of records - + + Defines TExtendedStrings which is a normal TStringList, except that the Objects property can hold any type of records + + + - - - - - + - - - - - + + - - - + + Defines actions needed when creating or freeing records in TExtendedStringList + + +

+ TExtStringsOption is an enumeration type with values that define actions needed when creating or freeing records in TExtendedStringList. Values from TExtStringsOption are stored in the TExtStringsOptions type. +

+
+
- + Memory allocated to a new record is cleared (filled with #0) - + Objects call their free method before they are deleted + - - - + + Stores TExtStringsOption values + + + TExtStringsOptions is a set type used to store TExtStringsOption values. TExtStringsOptions is the type used for the TExtendedStringList.Options property. + + + - - - - - + + + Implements a string list that allows Records in its Objects property + + +

+ TExtendedStringList is a TStringList descendant that provides support for using Records in the class. The Records property allows access to records stored in the class. TExtendedStringList provides overridden methods to maintain values in the Objects property. New properties and methods are included to control actions needed when adding or freeing Records stored in the Objects property. +

+
+
+ - - - - + - - - @@ -75,23 +83,19 @@ + - - - + - - - @@ -101,23 +105,19 @@ + - - - + - - - @@ -131,162 +131,297 @@ + - - - - + + Changes the memory allocation size for the specified record + + +

+ ResizeRecord is a procedure used to change the memory allocation size for the specified record. ResizeRecord calls ReAllocMem to change the allocated memory size to the value in NewSize. ResizeRecord is called when the value in the RecordSize property is changed. +

+
+
- + Pointer to the record - + Ordinal position for the specified record - + Previous memory allocation size - + New memory allocation size + - - - - + + Reimplements the read access specifier for Objects + + +

+ GetObject is an overridden TObject function which reimplements the read access specifier for the Objects property. GetObject calls the inherited method to retrieve the value at the position in Index. GetObject ensures that the return value is cast to a TObject instance when assigned, or Nil when unassigned. +

+

+ GetObject and PutObject are called directly (as opposed to using the Objects property) in methods which maintain values in Records. +

+
+
- + TObject stored at the specified position - + Ordinal position for the requested value + - - - - + + Reimplements the write access specifier for Objects + + +

+ PutObject is an overridden procedure which reimplements the write access specifier for the Objects property. PutObject uses a pointer to the value in Records stored at the position in Index. If a record has not been stored at the position, CreateRecord is called to allocate and initialize the memory for the record. The value in AnObject is stored in Records at the position in Index. +

+

+ GetObject and PutObject are called directly (as opposed to using the Objects property) in methods which maintain values in Records. +

+
+
- + Ordinal position for value - + Object to store at the specified position + - - - - + + Constructor for the class instance + + +

+ Create is the constructor for the class instance. Create calls the inherited constructor, and sets the default values in the following properties: +

+
+
Options
+
Set to [ esoClearRecordsOnCreate ]
+
RecordSize
+
Set to the value in the InitialRecordSize argument
+
+

+ Use RecordSize to change the memory allocation size for Records in the class. +

+
+
- + Memory size allocated for records + - - - - + + Destructor for the class instance + + +

+ Destroy is the destructor for the class instance. Destroy calls FreeAllRecords to release memory allocated for entries in the Records property. Destroy calls the inherited destructor. +

+
+
+ - - - - + + Removes items stored in the string list + + +

+ Clear is an overridden procedure used to remove items stored in the string list. Clear calls FreeAllRecords to release memory allocated for items in the Records property. Clear calls the inherited Clear method. +

+
+
+ - - - - + + Deletes the item at the specified position + + +

+ Delete is an overridden procedure used to delete the item at the specified position in the string list. Delete ensures that memory allocated in Records is freed by calling FreeRecord using the position in Index. Delete calls the inherited Delete method. +

+
+
- + Ordinal position for the item + - - - - + + Allocates memory for a new record + + +

+ CreateRecord is a procedure used to allocate (and optionally initialize) memory for a new value in the Records property. CreateRecord calls GetMem to allocate the memory size specified in the RecordSize property. When Options includes the value esoClearRecordsOnCreate, the allocated memory is filled with the character value #0. CreateRecord calls the inherited PutObject method to store the new record. +

+

+ CreateRecord is called when reading a value in the Records property, and when writing a value to the Objects property. +

+
+
- + Ordinal position for the record + - - - - + + Frees memory allocated for the specified record + + +

+ FreeRecord is a procedure used to free memory allocated for the record at the specified position. FreeRecord calls the inherited GetObject method to get the value stored in the Objects property. If a record has been allocated, and Options contains the value esoFreeObjectsOnDelete, the object instance in Objects is freed. FreeRecord calls FreeMem to release memory allocated for the record, and calls the inherited PutObject method to store the Nil'd object reference. +

+

+ FreeRecord is called when writing a new value in the Records property, and in methods like FreeAllRecords and Delete. +

+
+
- + Ordinal position for the record + - - - - + + Frees memory allocated for Records in the class + + +

+ FreeAllRecords is a procedure used to free memory allocated for the Records in the class. FreeAllRecords iterates over the items in the string list, and calls the FreeRecord method to free memory allocated for each item. +

+

+ FreeAllRecords is called in the Clear method, and in the destructor for the class. +

+
+ +
+ - - - - + + Determines if memory has been allocated for the specified record + + +

+ RecordAllocated is a Boolean function which determines if memory has been allocated for the record at the specified position. The return value is True if the item at the Index position in Objects contains a value other than Nil. +

+

+ RecordAllocated is called when reading a value for the Records property. +

+
+
- + True if memory has been allocated for the record - + Ordinal position for the record + - - - + + Provides indexed access to records in the string list + + +

+ Records an indexed Pointer property that provides access to records stored in the string list. Index is the ordinal position in the list for the desired record. +

+

+ Reading a value in Records causes a record to be created using CreateRecord if one has not already been allocated. The record pointer is retrieved by calling the inherited GetObject method. +

+

+ Writing a new value in Records causes a previous record value to be freed by calling FreeRecord. The inherited PutObject method is called to store the value in AValue as a TObject reference in the Objects property. +

+
+
- + Ordinal position for the record + - - - + + Memory allocation size for Records in the class + + +

+ RecordSize is an Integer property that specifies the memory allocation size for Records in the class. The initial value for the property is set to the argument passed in the constructor. Setting a new value for the property causes the allocated memory size for any existing Records to be changed to the new value. +

+
+
+ - - - + + Contains record options enabled in the class + + +

+ Options is a TExtStringsOptions property that contains the record options enabled for the class. Options can contains values from the TExtStringsOption enumeration, including: +

+
+
esoClearRecordsOnCreate
+
+ Memory allocated to a new record is cleared (filled with #0) +
+
esoFreeObjectsOnDelete
+
+ Objects call their free method before they are deleted +
+
+

+ The default value for the property is [ esoClearRecordsOnCreate ] as set in the constructor for the class. +

+
+
+
diff --git a/docs/xml/lazutils/html2textrender.xml b/docs/xml/lazutils/html2textrender.xml new file mode 100644 index 0000000000..092868af5d --- /dev/null +++ b/docs/xml/lazutils/html2textrender.xml @@ -0,0 +1,241 @@ + + + + + + + Contains an HTML-to-Text renderer + + +

+ html2textrender.pas contains an HTML-to-Text renderer. It converts HTML into plain text by stripping tags and their attributes. +

+
+ + + + + + + + + +

+ THTML2TextRenderer is an HTML-to-Text renderer. It converts HTML into plain text by stripping tags and their attributes. Converted text includes configurable indentation for HTML tags that affect the indentation level. The following HTML tags include special processing in the renderer: +

+
    +
  • HTML
  • +
  • BODY
  • +
  • P
  • +
  • BR
  • +
  • HR
  • +
  • OL
  • +
  • UL
  • +
  • LI
  • +
  • DIV CLASS="TITLE" (forces title mark output)
  • +
+

+ The following Named character entities are converted to their plain text equivalent: +

+
+
 
+
' '
+
<
+
'<'
+
&gt;
+
'>;'
+
&amp;
+
'&'
+
+

+ Other named character entities or numeric character entities are included verbatim in the plain text output. +

+

+ A UTF-8 Byte Order Mark in the HTML is ignored. +

+
+ +
+ + + + + + + + + + + + + End of line marker, by default standard LineEnding + + + Markup used at the start/end of title text + + + Markup used for an HR Tag + + + Markup used at the start of an Anchor Tag + + + Markup used at the end of an Anchor Tag + + + Markup used for a list item tag + + + Text added when there are too many lines + + + + + + + + + + + + + + + Increment (in spaces) for each nested HTML level + + + + + + + + + + + + + + + + Sets a pending line break to be added later + + + Sets a maximum of one pending line break to be added later + + + Appends text to the plaint-text output for the renderer + + + + + + + + + Handles an HTML tag and its attributes values + + + + + + Handles an HTML character entity + + + + + + Rest the state and output for the renderer + + + + + Creates the class instance + + + + + + + + + + + + Frees the class instance + + + + + + + Parses the HTML and renders to plain text + + +

+ Parses the HTML and renders to plain text. Output is limited to aMaxLines lines. Note: AddOutput, HtmlTag and HtmlEntity return False if MaxLines was exceeded. +

+
+ +
+ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
+ +
+
diff --git a/docs/xml/lazutils/uitypes.xml b/docs/xml/lazutils/uitypes.xml new file mode 100644 index 0000000000..93c734be73 --- /dev/null +++ b/docs/xml/lazutils/uitypes.xml @@ -0,0 +1,262 @@ + + + + + + + Contains types and constants used in modal dialogs + + + + + + + + + Defines the type of dialog implemented in a message box + + +

+ TMsgDlgType is an enumeration type that defines the type of dialog implemented in a message box. Values in the enumeration determine the caption, icon, string literals, and buttons displayed in various dialog types. +

+

+ TMsgDlgType is used in several routines, such as: MessageDlg, MessageDlgPos, MessageDlgPosHelp, CreateMessageDialog, QuestionDlg, and LazMessageDlg. +

+
+ +
+ + + Warning message box with an Exclamation mark icon + + + + + Error message box with Stop sign icon + + + + + Information message box with an Info icon + + + + + + Confirmation message box with a Question mark icon + + + + + Custom message box; No icon is displayed, and the caption contains the value in ApplicationName + + + + + + Defines the buttons that can be displayed in a message box + + +

+ TMsgDlgBtn is an enumeration type with values that define the buttons that can be displayed in a message dialog. TMsgDlgBtn values also determine the modal result returned from the dialog when the corresponding button is clicked. +

+

+ Values in the enumeration are used in various routines, such as: MessageDlg, MessageDlgPos, MessageDlgPosHelp, CreateMessageDialog, QuestionDlg, and LazMessageDlg. +

+
+ +
+ + + Displays a 'Yes' button + + + + + Displays a 'No' button + + + + + Displays an 'OK' button + + + + + Displays a 'Cancel' button + + + + + Displays an 'Abort' button + + + + + Displays a 'Retry' button + + + + + Displays an 'Ignore' button + + + + + Displays an 'All' button + + + + + Displays a 'No to All' button + + + + + Displays a 'Yes to All' button + + + + + Displays a 'Help' button + + + + + Displays a 'Close' button + + + + + + Stores values from the TMsgDlgBtn enumeration + + +

+ TMsgDlgButtons is a set type used to store zero or more values from the TMsgDlgBtn enumeration. TMsgDlgButtons is passed as an argument to routines like: MessageDlg, MessageDlgPos, MessageDlgPosHelp, CreateMessageDialog, QuestionDlg, and LazMessageDlg. +

+
+ +
+ + + + Value returned from a modal dialog box + + +

+ TModalResult is an Integer type with a value in the range low(Integer)..high(Integer). TModalResult is the type returned by routines that display modal dialog boxes, such as: MessageDlg, MessageDlgPos, MessageDlgPosHelp, QuestionDlg, and LazMessageDlg. +

+

+ The value in TModalResult indicates the button clicked to close the modal dialog, and contains one of the constant values: +

+
    +
  • mrNone
  • +
  • mrOK
  • +
  • mrCancel
  • +
  • mrAbort
  • +
  • mrRetry
  • +
  • mrIgnore
  • +
  • mrYes
  • +
  • mrNo
  • +
  • mrAll
  • +
  • mrNoToAll
  • +
  • mrYesToAll
  • +
  • mrClose
  • +
  • mrLast
  • +
+
+ +
+ + + + Pointer to a modal result value + + +

+ PModalResult is a pointer type to a TModalResult value. +

+
+ +
+ + + Dialog was closed without pressing a button + + + OK button was clicked + + + Cancel button was clicked + + + Abort button was clicked + + + Retry button was clicked + + + Ignore button was clicked + + + Yes button was clicked + + + No button was clicked + + + All button was clicked + + + No to All button was clicked + + + Yes to All button was clicked + + + Close button was clicked + + + Last modal result value (same as mrClose) + + + + + String representations for modal result values + + +

+ ModalResultStr is array of ShortString values that contains the string representations for values in TModalResult. ModalResultStr values can be accessed using the corresponding numeric modal result value. For example: +

+ + sResult := ModalResultStr[mrNone]; // returns 'mrNone' + +

+ ModalResultStr contains the following values: +

+
    +
  • 'mrNone'
  • +
  • 'mrOk'
  • +
  • 'mrCancel'
  • +
  • 'mrAbort'
  • +
  • 'mrRetry'
  • +
  • 'mrIgnore'
  • +
  • 'mrYes'
  • +
  • 'mrNo'
  • +
  • 'mrAll'
  • +
  • 'mrNoToAll'
  • +
  • 'mrYesToAll'
  • +
  • 'mrClose'
  • +
+
+ +
+
+ +
+
diff --git a/docs/xml/lcl/pairsplitter.xml b/docs/xml/lcl/pairsplitter.xml index aad40e89ce..7b597038ea 100644 --- a/docs/xml/lcl/pairsplitter.xml +++ b/docs/xml/lcl/pairsplitter.xml @@ -2,574 +2,493 @@ + ==================================================================== + PairSplitter + ==================================================================== + --> - - + + Defines the TPairSplitter component + + + Defines the TPairSplitter component. TPairSplitter is a component with two TPairSplitterSide children. Both child components can contain other components. The children are separated by a splitter which can be dragged by the user to resize the adjacent children. + + - - - - - + - - - - - + - - - - - + - - - - - + - - - - - + - - - - - + - - - - - + - - - - - + - - - - - + + - - - - - - - - - - - - - - - - - + + + Implements the child controls used in the TPairSplitter component. + + +

+ TPairSplitterSide is a TWinControl descendant that implements the child controls used in the TPairSplitter component. TPairSplitterSide behaves like a TPanel component with a reference to the TPairSplitter that owns the control. TPairSplitterSide provides a custom Paint method and WMPaint message handling. TPairSplitterSide alters the visibility of inherited properties to match their intended usage. +

+
+ + + - - - - + Gets the value for the Splitter property - + Value for the property + + + - - - + + Sets the value for the Parent property + + +

+ SetParent is an overridden procedure used to set the value in the Parent property. SetParent ensures that the control is removed from its existing parent prior to setting the new property value. The control is added to the new parent when AParent is not Nil. +

+

+ SetParent is the reimplemented write access specifier for the Parent property. +

+
- + New value for the Parent property + - - - - + + Provides custom handling for WMPaint messages + - + WM_PAINT message for the control + + + - - - - - - - - - - - - - - - + Renders the control on its Canvas + + + + + - - - + + Constructor for the class instance + + +

+ Create is the constructor for the class instance, and calls the inherited constructor using TheOwner as the owner of the control. Create updates the ControlStyle property to include the value csAcceptsControls. +

+
- + Owner of the control + - - - + + Destructor for the class instance + + +

+ Destroy if the destructor for the class instance, and calls the inherited destructor. +

+
+ - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - + + TPairSplitter for the control + + +

+ Splitter is a read-only TCustomPairSplitter property which represents the component for the control. The value in Splitter is the same as the Parent for the control, or Nil when Parent has not been assigned. +

+
+ + + + + + + + + + + + + + + + + + + + + + + + - - + + Sets the orientation for the Splitter in TPairSplitter + + +

+ TPairSplitterType is an enumeration type with values that set the orientation for the splitter in the TPairSplitter component. The values refer to the splitter orientation relative to the adjacent children. +

+

+ TPairSplitterType is the type used for the TCustomPairSplitter.SplitterType property. +

+
- + Splitter is oriented to size adjacent children horizontally - + Splitter is oriented to size adjacent children vertically + - - - - + + + Defines the base class for the TPairSplitter component + + + Defines the base class for the TPairSplitter component. TPairSplitter is a component with two TPairSplitterSide children. Both child components can contain other components. The children are separated by a splitter which can be dragged by the user to resize the adjacent children. + + - - - + + + + + + + + + + + + + + + + + + + + + + Gets the value for the Cursor property + + + Value for the Cursor property + + + + Sets the value for the Cursor property + + + New value for the Cursor property + + + + + Gets the default size for the control class + + +

+ GetControlClassDefaultSize is a TSize class function used to get the default size for instances of the class. GetControlClassDefaultSize sets the Height and Width in the TSize value. The default value in both CX and CY is 90 pixels. +

+

+ GetControlClassDefaultSize is used the Create constructor. +

+
- - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - + + + TSize type with the default Height and Width for instances of the class + + - - - + + Constructor for the class instance + + +

+ Create is the constructor for the class instance. Create calls the inherited constructor using TheOwner as the owner of the component. +

+

+ Create calls the GetControlClassDefaultSize class method to get the default height and width values used in instances of the class type, and calls SetBounds to apply the values. Create sets the default values for the following properties: +

+
+
ControlStyle
+
Removes the value csAcceptsControls in the property
+
SplitterType
+
pstHorizontal
+
Cursor
+
crHSplit
+
Position
+
45
+
+

+ Create calls the CreateSides method when csDesigning is not in the ComponentState property. CreateSides ensures both TPairSplitterSide instances are created (and parented) in the Sides property. +

+
- + Owner of the component + - - - + + Destructor for the class instance + + +

+ Destroy is the destructor for the class instance. Destroy frees resources allocated in the Sides property, and calls the inherited destructor. +

+
+ - - - + + Creates the window handle for the control + + +

+ CreateWnd is an overridden procedure used to create the window handle for the control. CreateWnd calls CreateSides to ensure both TPairSplitterSide instances are created (and parented) in the Sides property. CreateWnd calls the inherited CreateWnd method. +

+
+ - - - + + Updates the value in the Position property + + + UpdatePosition is a procedure used to update the value in the Position property. UpdatePosition calls the SetPosition method in the WidgetSetClass to recalculate the current relative position for the Splitter in the component. UpdatePosition stores the value in the Position property. + + - - - + + Creates missing controls in the Sides property + + +

+ CreateSides is a procedure used to create missing controls in the Sides property. CreateSides ensures that both TPairSplitterSide values used in the component have been created (and parented). +

+

+ No actions are performed in the method when component streaming has not been completed, or when the component is freed. +

+

+ CreateSides is called from the Create, CreateWnd, and Loaded methods. +

+
+ - - - + + Performs actions on completion of component streaming + + +

+ Loaded is an overridden procedure which performs actions needed when the component has been loaded using the LCL component streaming mechanism. Loaded calls the inherited Loaded method. Loaded calls the CreateSides method to ensure that both values in the Sides property have been created. Loaded updates the value in Position if the handle for the control has been allocated. +

+
- - - - - - - - - - - - + - - - + + Determines if the class is inherited from a specific ancestor + + +

+ ChildClassAllowed is an overridden Boolean function used to determine if the specified class is inherited from a specific ancestor class. ChildClassAllowed calls the ChildClass.InheritsFrom method. The return value is True when it is derived from the TPairSplitterSide or TSplitter classes. +

+
- - - - - - + Class to compare in the method` + + True if the class has a specific ancestor + + + + Cursor displayed when the control is resized + + - - + + Provides access to the adjacent children resized using the Splitter + + +

+ Sides is a read-only indexed TPairSplitterSide property which provides access to the adjacent children resized using the Splitter in the component. Sides always has 2 elements in the array (in the range 0..1). If Index contains an invalid value, an exception is raised. +

+

+ Each controls in Sides behaves like a TPanel component with a reference to the TPairSplitter that owns the control. They provide a custom Paint method and WMPaint message handling. Some of their properties have altered visibilities; for example, Align and Anchors are private, Top and Left are public. +

+
+ +

+ Calls RaiseGDBException to raise an exception when Index < 0 or Index > 1. Raised with the message 'TCustomPairSplitter.GetSides: Index out of bounds'. +

+
- + Ordinal position the adjacent child control + - - + + Specifies the orientation for the splitter in the component + + +

+ SplitterType is a TPairSplitterType property which specifies the orientation for the splitter in the component. The values in TPairSplitterType dictate the splitter orientation relative to the adjacent children in Sides. Writing a new value in the SplitterType property causes the value in the Cursor property to be changed to match the selected orientation. For example: +

+
+
pstHorizontal
+
+ Splitter is oriented to size adjacent children horizontally. Cursor is set to crHSplit. (Default value) +
+
pstVertical
+
+ Splitter is oriented to size adjacent children vertically. Cursor is set to crVSplit. +
+ +
+ - - + + Indicates the position for the Splitter relative to its orientation + + +

+ Position is an Integer property that indicates the position for the splitter relative to its orientation. Reading the value in splitter causes the UpdatePosition to be called to recalculate the current position for the Splitter. Position can be set at design-time, or changed at run-time by dragging the grab bar for the splitter. The value in Position affects the sizing of the adjacent child controls in Sides. +

+

+ When SplitterType contains pstHorizontal, the value in Position is used as the Width for the child control in Sides[0]. The remaining Width in the component (less the 5 pixels used for the Splitter) is set as the Width for the child control in Sides[1]. +

+

+ When SplitterType contains pstVertical, the value in Position is used as the Height for the child control in Sides[0]. The remaining Height in the component (less the 5 pixels used for the Splitter) is set as the Height for the child control in Sides[1]. +

+
+ - - - - + + + Implements the TPairSplitter component + + + Implements the TPairSplitter component. TPairSplitter is a component with two TPairSplitterSide children. Both child components can contain other components. The children are separated by a splitter which can be dragged by the user to resize the adjacent children. + + - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - + + + + + + + + + + + + + + + + + + + + + + + + - - - - + - -