mirror of
https://github.com/greenshot/greenshot
synced 2025-07-11 07:37:05 -07:00
git-svn-id: http://svn.code.sf.net/p/greenshot/code/trunk@704 7dccd23d-a4a3-4e1f-8c07-b4c1b4018ab4
299 lines
8.7 KiB
C#
299 lines
8.7 KiB
C#
/*
|
|
* Greenshot - a free and open source screenshot tool
|
|
* Copyright (C) 2007-2010 Thomas Braun, Jens Klingen, Robin Krom
|
|
*
|
|
* For more information see: http://getgreenshot.org/
|
|
* The Greenshot project is hosted on Sourceforge: http://sourceforge.net/projects/greenshot/
|
|
*
|
|
* 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 1 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, see <http://www.gnu.org/licenses/>.
|
|
*/
|
|
using System;
|
|
using System.Collections.Generic;
|
|
using System.ComponentModel;
|
|
using System.Drawing;
|
|
using System.Drawing.Imaging;
|
|
using System.Drawing.Text;
|
|
using System.Runtime.Serialization;
|
|
using System.Windows.Forms;
|
|
|
|
using Greenshot.Drawing.Fields;
|
|
using Greenshot.Helpers;
|
|
|
|
namespace Greenshot.Drawing {
|
|
/// <summary>
|
|
/// Represents a textbox (extends RectangleContainer for border/background support
|
|
/// </summary>
|
|
[Serializable()]
|
|
public class TextContainer : RectangleContainer, ITextContainer {
|
|
|
|
private bool fontInvalidated = true;
|
|
private Font font;
|
|
|
|
private string text;
|
|
public string Text {
|
|
get { return text; }
|
|
set {
|
|
if((text == null && value != null) || !text.Equals(value)) {
|
|
text = value;
|
|
OnPropertyChanged("Text");
|
|
}
|
|
}
|
|
}
|
|
|
|
[NonSerialized]
|
|
private TextBox textBox;
|
|
|
|
public TextContainer(Surface parent) : base(parent) {
|
|
Init();
|
|
Field field = FieldFactory.CreateField(FieldType.FONT_BOLD);
|
|
//field.PropertyChanged += FontPropertyChanged;
|
|
AddField(field);
|
|
AddField(FieldFactory.CreateField(FieldType.FONT_ITALIC));
|
|
AddField(FieldFactory.CreateField(FieldType.FONT_FAMILY));
|
|
AddField(FieldFactory.CreateField(FieldType.FONT_SIZE));
|
|
AddField(FieldFactory.CreateField(FieldType.LINE_COLOR, GetType()));
|
|
AddField(FieldFactory.CreateField(FieldType.LINE_THICKNESS, GetType()));
|
|
AddField(FieldFactory.CreateField(FieldType.FILL_COLOR, GetType()));
|
|
AddField(FieldFactory.CreateField(FieldType.SHADOW));
|
|
}
|
|
|
|
[OnDeserializedAttribute()]
|
|
private void OnDeserialized(StreamingContext context) {
|
|
Init();
|
|
UpdateFont();
|
|
}
|
|
|
|
/**
|
|
* Destructor
|
|
*/
|
|
~TextContainer() {
|
|
Dispose(false);
|
|
}
|
|
|
|
/**
|
|
* The public accessible Dispose
|
|
* Will call the GarbageCollector to SuppressFinalize, preventing being cleaned twice
|
|
*/
|
|
public override void Dispose() {
|
|
Dispose(true);
|
|
base.Dispose();
|
|
GC.SuppressFinalize(this);
|
|
}
|
|
|
|
/**
|
|
* This Dispose is called from the Dispose and the Destructor.
|
|
* When disposing==true all non-managed resources should be freed too!
|
|
*/
|
|
protected virtual void Dispose(bool disposing) {
|
|
if (disposing) {
|
|
if(textBox != null) textBox.Dispose();
|
|
if(font != null) font.Dispose();
|
|
}
|
|
textBox = null;
|
|
font = null;
|
|
}
|
|
|
|
private void Init() {
|
|
CreateTextBox();
|
|
this.PropertyChanged += new PropertyChangedEventHandler(TextContainer_PropertyChanged);
|
|
this.FieldChanged += new FieldChangedEventHandler(TextContainer_FieldChanged);
|
|
}
|
|
|
|
public void FitToText() {
|
|
UpdateFont();
|
|
Size textSize = TextRenderer.MeasureText(text, font);
|
|
int lineThickness = GetFieldValueAsInt(FieldType.LINE_THICKNESS);
|
|
Width = textSize.Width + lineThickness;
|
|
Height = textSize.Height + lineThickness;
|
|
}
|
|
|
|
void TextContainer_PropertyChanged(object sender, PropertyChangedEventArgs e) {
|
|
if(e.PropertyName.Equals("Selected")) {
|
|
if(!Selected && textBox.Visible) HideTextBox();
|
|
else if(Selected && Status==EditStatus.DRAWING) {
|
|
ShowTextBox();
|
|
}
|
|
//else if(!textBox.Visible) ShowTextBox();
|
|
}
|
|
if(textBox.Visible) {
|
|
UpdateTextBoxPosition();
|
|
UpdateTextBoxFormat();
|
|
textBox.Invalidate();
|
|
}
|
|
}
|
|
|
|
void TextContainer_FieldChanged(object sender, FieldChangedEventArgs e)
|
|
{
|
|
if(textBox.Visible) {
|
|
UpdateTextBoxFormat();
|
|
textBox.Invalidate();
|
|
} else {
|
|
UpdateFont();
|
|
//Invalidate();
|
|
}
|
|
font.Dispose();
|
|
font = null;
|
|
fontInvalidated = true;
|
|
}
|
|
|
|
public override void OnDoubleClick() {
|
|
ShowTextBox();
|
|
}
|
|
|
|
private void CreateTextBox() {
|
|
textBox = new TextBox();
|
|
textBox.Multiline = true;
|
|
textBox.AcceptsTab = false;
|
|
textBox.AcceptsReturn = false;
|
|
textBox.DataBindings.Add("Text", this, "Text", false, DataSourceUpdateMode.OnPropertyChanged);
|
|
textBox.LostFocus += new EventHandler(textBox_LostFocus);
|
|
textBox.KeyDown += new KeyEventHandler(textBox_KeyDown);
|
|
textBox.BorderStyle = BorderStyle.FixedSingle;
|
|
textBox.Visible = false;
|
|
}
|
|
|
|
private void ShowTextBox() {
|
|
childLabel.Hide();
|
|
parent.KeysLocked = true;
|
|
parent.Controls.Add(textBox);
|
|
textBox.Show();
|
|
textBox.Focus();
|
|
}
|
|
|
|
private void HideTextBox() {
|
|
childLabel.Show();
|
|
parent.Focus();
|
|
textBox.Hide();
|
|
parent.KeysLocked = false;
|
|
parent.Controls.Remove(textBox);
|
|
}
|
|
|
|
private void UpdateFont() {
|
|
string fontFamily = GetFieldValueAsString(FieldType.FONT_FAMILY);
|
|
bool fontBold = GetFieldValueAsBool(FieldType.FONT_BOLD);
|
|
bool fontItalic = GetFieldValueAsBool(FieldType.FONT_ITALIC);
|
|
float fontSize = GetFieldValueAsFloat(FieldType.FONT_SIZE);
|
|
|
|
if(fontInvalidated && fontFamily != null && fontSize != 0) {
|
|
FontStyle fs = FontStyle.Regular;
|
|
|
|
bool hasStyle = false;
|
|
using(FontFamily fam = new FontFamily(fontFamily)) {
|
|
bool boldAvailable = fam.IsStyleAvailable(FontStyle.Bold);
|
|
if(fontBold && boldAvailable) {
|
|
fs |= FontStyle.Bold;
|
|
hasStyle = true;
|
|
}
|
|
|
|
bool italicAvailable = fam.IsStyleAvailable(FontStyle.Italic);
|
|
if(fontItalic && italicAvailable) {
|
|
fs |= FontStyle.Italic;
|
|
hasStyle = true;
|
|
}
|
|
|
|
if(!hasStyle) {
|
|
bool regularAvailable = fam.IsStyleAvailable(FontStyle.Regular);
|
|
if(regularAvailable) {
|
|
fs = FontStyle.Regular;
|
|
} else {
|
|
if(boldAvailable) {
|
|
fs = FontStyle.Bold;
|
|
} else if(italicAvailable) {
|
|
fs = FontStyle.Italic;
|
|
}
|
|
}
|
|
}
|
|
|
|
font = new Font(fam, fontSize, fs, GraphicsUnit.Pixel);
|
|
}
|
|
fontInvalidated = false;
|
|
|
|
}
|
|
}
|
|
|
|
private void UpdateTextBoxPosition() {
|
|
textBox.Left = this.Left;
|
|
textBox.Top = this.Top;
|
|
textBox.Width = this.Width;
|
|
textBox.Height = this.Height;
|
|
|
|
}
|
|
|
|
private void UpdateTextBoxFormat() {
|
|
UpdateFont();
|
|
Color lineColor = GetFieldValueAsColor(FieldType.LINE_COLOR);
|
|
textBox.ForeColor = lineColor;
|
|
textBox.Font = font;
|
|
}
|
|
|
|
void textBox_KeyDown(object sender, KeyEventArgs e) {
|
|
// ESC and Enter/Return (w/o Shift) hide text editor
|
|
if(e.KeyCode == Keys.Escape || ((e.KeyCode == Keys.Return || e.KeyCode == Keys.Enter) && e.Modifiers != Keys.Shift)) {
|
|
HideTextBox();
|
|
e.SuppressKeyPress = true;
|
|
}
|
|
}
|
|
|
|
void textBox_LostFocus(object sender, EventArgs e) {
|
|
HideTextBox();
|
|
}
|
|
|
|
public override void Draw(Graphics g, RenderMode rm) {
|
|
base.Draw(g, rm);
|
|
UpdateFont();
|
|
|
|
Rectangle rect = GuiRectangle.GetGuiRectangle(this.Left, this.Top, this.Width, this.Height);
|
|
if (Selected && !rm.Equals(RenderMode.EXPORT)) {
|
|
DrawSelectionBorder(g, rect);
|
|
}
|
|
|
|
if (text == null || text.Length == 0 ) {
|
|
return;
|
|
}
|
|
|
|
// we only draw the shadow if there is no background
|
|
bool shadow = GetFieldValueAsBool(FieldType.SHADOW);
|
|
Color fillColor = GetFieldValueAsColor(FieldType.FILL_COLOR);
|
|
int lineThickness = GetFieldValueAsInt(FieldType.LINE_THICKNESS);
|
|
int textOffset = (lineThickness>0) ? (int)Math.Ceiling(lineThickness/2d) : 0;
|
|
// draw shadow before anything else
|
|
if ( shadow && (fillColor == Color.Transparent || fillColor == Color.Empty)) {
|
|
int basealpha = 100;
|
|
int alpha = basealpha;
|
|
int steps = 5;
|
|
int currentStep = 1;
|
|
while (currentStep <= steps) {
|
|
int offset = currentStep;
|
|
Rectangle shadowRect = GuiRectangle.GetGuiRectangle(Left + offset, Top + offset, Width, Height);
|
|
if(lineThickness>0) shadowRect.Inflate(-textOffset, -textOffset);
|
|
using (Brush fontBrush = new SolidBrush(Color.FromArgb(alpha, 100, 100, 100))) {
|
|
g.DrawString(text, font, fontBrush, shadowRect);
|
|
currentStep++;
|
|
alpha = alpha - basealpha / steps;
|
|
}
|
|
}
|
|
}
|
|
|
|
Color lineColor = GetFieldValueAsColor(FieldType.LINE_COLOR);
|
|
Rectangle fontRect = rect;
|
|
if(lineThickness>0) fontRect.Inflate(-textOffset,-textOffset);
|
|
using (Brush fontBrush = new SolidBrush(lineColor)) {
|
|
g.DrawString(text, font, fontBrush, fontRect);
|
|
}
|
|
|
|
}
|
|
|
|
|
|
}
|
|
}
|