mirror of
https://github.com/aseprite/aseprite.git
synced 2025-04-04 10:20:14 +00:00
The UndoHistory was simplified, m_enabled/label/modification members were removed because they aren't part of the undo library goal. Now the "label" field is part of Open/CloseGroup undoers, and all undoers must be enclosed in a group (this restriction is only for the app, the undo library doesn't have this restriction). In this way, the label/modification information is exchanged between an OpenGroup <-> CloseGroup pair only.
876 lines
25 KiB
C++
876 lines
25 KiB
C++
/* ASEPRITE
|
|
* Copyright (C) 2001-2012 David Capello
|
|
*
|
|
* 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
|
|
*/
|
|
|
|
#include "config.h"
|
|
|
|
#include "app.h"
|
|
#include "app/color.h"
|
|
#include "app/color_utils.h"
|
|
#include "app/file_selector.h"
|
|
#include "base/bind.h"
|
|
#include "base/compiler_specific.h"
|
|
#include "commands/command.h"
|
|
#include "commands/params.h"
|
|
#include "console.h"
|
|
#include "document_undo.h"
|
|
#include "document_wrappers.h"
|
|
#include "gfx/hsv.h"
|
|
#include "gfx/rgb.h"
|
|
#include "gfx/size.h"
|
|
#include "ini_file.h"
|
|
#include "modules/editors.h"
|
|
#include "modules/gui.h"
|
|
#include "modules/palettes.h"
|
|
#include "raster/image.h"
|
|
#include "raster/palette.h"
|
|
#include "raster/quantization.h"
|
|
#include "raster/sprite.h"
|
|
#include "raster/stock.h"
|
|
#include "skin/skin_slider_property.h"
|
|
#include "ui/graphics.h"
|
|
#include "ui/gui.h"
|
|
#include "ui_context.h"
|
|
#include "undo_transaction.h"
|
|
#include "undoers/set_palette_colors.h"
|
|
#include "widgets/color_bar.h"
|
|
#include "widgets/color_sliders.h"
|
|
#include "widgets/editor/editor.h"
|
|
#include "widgets/hex_color_entry.h"
|
|
#include "widgets/palette_view.h"
|
|
#include "widgets/status_bar.h"
|
|
|
|
#include <allegro.h>
|
|
#include <stdio.h>
|
|
#include <string.h>
|
|
#include <vector>
|
|
|
|
using namespace gfx;
|
|
using namespace ui;
|
|
|
|
class PaletteEntryEditor : public Frame
|
|
{
|
|
public:
|
|
PaletteEntryEditor();
|
|
~PaletteEntryEditor();
|
|
|
|
void setColor(const Color& color);
|
|
|
|
protected:
|
|
bool onProcessMessage(Message* msg) OVERRIDE;
|
|
|
|
void onExit();
|
|
void onCloseFrame();
|
|
void onFgBgColorChange(const Color& color);
|
|
void onColorSlidersChange(ColorSlidersChangeEvent& ev);
|
|
void onColorHexEntryChange(const Color& color);
|
|
void onColorTypeButtonClick(Event& ev);
|
|
void onMoreOptionsClick(Event& ev);
|
|
void onCopyColorsClick(Event& ev);
|
|
void onPasteColorsClick(Event& ev);
|
|
void onLoadPaletteClick(Event& ev);
|
|
void onSavePaletteClick(Event& ev);
|
|
void onRampClick(Event& ev);
|
|
void onQuantizeClick(Event& ev);
|
|
|
|
private:
|
|
void selectColorType(Color::Type type);
|
|
void setPaletteEntry(const Color& color);
|
|
void setPaletteEntryChannel(const Color& color, ColorSliders::Channel channel);
|
|
void setNewPalette(Palette* palette, const char* operationName);
|
|
void updateCurrentSpritePalette(const char* operationName);
|
|
void updateColorBar();
|
|
void onPalChange();
|
|
|
|
Box m_vbox;
|
|
Box m_topBox;
|
|
Box m_bottomBox;
|
|
RadioButton m_rgbButton;
|
|
RadioButton m_hsvButton;
|
|
HexColorEntry m_hexColorEntry;
|
|
Label m_entryLabel;
|
|
Button m_moreOptions;
|
|
RgbSliders m_rgbSliders;
|
|
HsvSliders m_hsvSliders;
|
|
Button m_copyButton;
|
|
Button m_pasteButton;
|
|
Button m_loadButton;
|
|
Button m_saveButton;
|
|
Button m_rampButton;
|
|
Button m_quantizeButton;
|
|
|
|
// This variable is used to avoid updating the m_hexColorEntry text
|
|
// when the color change is generated from a
|
|
// HexColorEntry::ColorChange signal. In this way we don't override
|
|
// what the user is writting in the text field.
|
|
bool m_disableHexUpdate;
|
|
|
|
ui::Timer m_redrawTimer;
|
|
bool m_redrawAll;
|
|
|
|
// True if the palette change must be implant in the UndoHistory
|
|
// (e.g. when two or more changes in the palette are made in short
|
|
// time).
|
|
bool m_implantChange;
|
|
|
|
// True if the PaletteChange signal is generated by the same
|
|
// PaletteEntryEditor instance.
|
|
bool m_selfPalChange;
|
|
|
|
Signal0<void>::SlotType* m_palChangeSlot;
|
|
|
|
// Internal-clipboard to copy & paste colors between palettes. It's
|
|
// used in onCopy/PasteColorsClick.
|
|
std::vector<uint32_t> m_clipboardColors;
|
|
};
|
|
|
|
//////////////////////////////////////////////////////////////////////
|
|
// PaletteEditorCommand
|
|
|
|
static PaletteEntryEditor* g_frame = NULL;
|
|
|
|
class PaletteEditorCommand : public Command
|
|
{
|
|
public:
|
|
PaletteEditorCommand();
|
|
Command* clone() { return new PaletteEditorCommand(*this); }
|
|
|
|
protected:
|
|
void onLoadParams(Params* params) OVERRIDE;
|
|
void onExecute(Context* context) OVERRIDE;
|
|
|
|
private:
|
|
bool m_open;
|
|
bool m_close;
|
|
bool m_switch;
|
|
bool m_background;
|
|
};
|
|
|
|
PaletteEditorCommand::PaletteEditorCommand()
|
|
: Command("PaletteEditor",
|
|
"PaletteEditor",
|
|
CmdRecordableFlag)
|
|
{
|
|
m_open = true;
|
|
m_close = false;
|
|
m_switch = false;
|
|
m_background = false;
|
|
}
|
|
|
|
void PaletteEditorCommand::onLoadParams(Params* params)
|
|
{
|
|
std::string target = params->get("target");
|
|
if (target == "foreground") m_background = false;
|
|
else if (target == "background") m_background = true;
|
|
|
|
std::string open_str = params->get("open");
|
|
if (open_str == "true") m_open = true;
|
|
else m_open = false;
|
|
|
|
std::string close_str = params->get("close");
|
|
if (close_str == "true") m_close = true;
|
|
else m_close = false;
|
|
|
|
std::string switch_str = params->get("switch");
|
|
if (switch_str == "true") m_switch = true;
|
|
else m_switch = false;
|
|
}
|
|
|
|
void PaletteEditorCommand::onExecute(Context* context)
|
|
{
|
|
// If this is the first time the command is execute...
|
|
if (!g_frame) {
|
|
// If the command says "Close the palette editor" and it is not
|
|
// created yet, we just do nothing.
|
|
if (m_close)
|
|
return;
|
|
|
|
// If this is "open" or "switch", we have to create the frame.
|
|
g_frame = new PaletteEntryEditor();
|
|
}
|
|
// If the frame is already created and it's visible, close it (only in "switch" or "close" modes)
|
|
else if (g_frame->isVisible() && (m_switch || m_close)) {
|
|
// Hide the frame
|
|
g_frame->closeWindow(NULL);
|
|
return;
|
|
}
|
|
|
|
if (m_switch || m_open) {
|
|
if (!g_frame->isVisible()) {
|
|
// Default bounds
|
|
g_frame->remap_window();
|
|
|
|
int width = MAX(jrect_w(g_frame->rc), JI_SCREEN_W/2);
|
|
g_frame->setBounds(Rect(JI_SCREEN_W - width - jrect_w(app_get_toolbar()->rc),
|
|
JI_SCREEN_H - jrect_h(g_frame->rc) - jrect_h(app_get_statusbar()->rc),
|
|
width, jrect_h(g_frame->rc)));
|
|
|
|
// Load window configuration
|
|
load_window_pos(g_frame, "PaletteEditor");
|
|
}
|
|
|
|
// Run the frame in background.
|
|
g_frame->open_window_bg();
|
|
app_get_colorbar()->setPaletteEditorButtonState(true);
|
|
}
|
|
|
|
// Show the specified target color
|
|
{
|
|
Color color =
|
|
(m_background ? context->getSettings()->getBgColor():
|
|
context->getSettings()->getFgColor());
|
|
|
|
g_frame->setColor(color);
|
|
}
|
|
}
|
|
|
|
//////////////////////////////////////////////////////////////////////
|
|
// PaletteEntryEditor implementation
|
|
//
|
|
// Based on ColorSelector class.
|
|
|
|
PaletteEntryEditor::PaletteEntryEditor()
|
|
: Frame(false, "Palette Editor (F4)")
|
|
, m_vbox(JI_VERTICAL)
|
|
, m_topBox(JI_HORIZONTAL)
|
|
, m_bottomBox(JI_HORIZONTAL)
|
|
, m_rgbButton("RGB", 1, JI_BUTTON)
|
|
, m_hsvButton("HSB", 1, JI_BUTTON)
|
|
, m_entryLabel("")
|
|
, m_moreOptions("+")
|
|
, m_copyButton("Copy")
|
|
, m_pasteButton("Paste")
|
|
, m_loadButton("Load")
|
|
, m_saveButton("Save")
|
|
, m_rampButton("Ramp")
|
|
, m_quantizeButton("Quantize")
|
|
, m_disableHexUpdate(false)
|
|
, m_redrawAll(false)
|
|
, m_implantChange(false)
|
|
, m_selfPalChange(false)
|
|
, m_redrawTimer(250, this)
|
|
{
|
|
m_topBox.setBorder(gfx::Border(0));
|
|
m_topBox.child_spacing = 0;
|
|
m_bottomBox.setBorder(gfx::Border(0));
|
|
|
|
setup_mini_look(&m_rgbButton);
|
|
setup_mini_look(&m_hsvButton);
|
|
setup_mini_look(&m_moreOptions);
|
|
setup_mini_look(&m_copyButton);
|
|
setup_mini_look(&m_pasteButton);
|
|
setup_mini_look(&m_loadButton);
|
|
setup_mini_look(&m_saveButton);
|
|
setup_mini_look(&m_rampButton);
|
|
setup_mini_look(&m_quantizeButton);
|
|
|
|
// Top box
|
|
m_topBox.addChild(&m_rgbButton);
|
|
m_topBox.addChild(&m_hsvButton);
|
|
m_topBox.addChild(&m_hexColorEntry);
|
|
m_topBox.addChild(&m_entryLabel);
|
|
m_topBox.addChild(new BoxFiller);
|
|
m_topBox.addChild(&m_moreOptions);
|
|
|
|
// Bottom box
|
|
{
|
|
Box* box = new Box(JI_HORIZONTAL);
|
|
box->child_spacing = 0;
|
|
box->addChild(&m_copyButton);
|
|
box->addChild(&m_pasteButton);
|
|
m_bottomBox.addChild(box);
|
|
}
|
|
{
|
|
Box* box = new Box(JI_HORIZONTAL);
|
|
box->child_spacing = 0;
|
|
box->addChild(&m_loadButton);
|
|
box->addChild(&m_saveButton);
|
|
m_bottomBox.addChild(box);
|
|
}
|
|
m_bottomBox.addChild(&m_rampButton);
|
|
m_bottomBox.addChild(&m_quantizeButton);
|
|
|
|
// Main vertical box
|
|
m_vbox.addChild(&m_topBox);
|
|
m_vbox.addChild(&m_rgbSliders);
|
|
m_vbox.addChild(&m_hsvSliders);
|
|
m_vbox.addChild(&m_bottomBox);
|
|
addChild(&m_vbox);
|
|
|
|
// Hide (or show) the "More Options" depending the saved value in .cfg file
|
|
m_bottomBox.setVisible(get_config_bool("PaletteEditor", "ShowMoreOptions", false));
|
|
|
|
m_rgbButton.Click.connect(&PaletteEntryEditor::onColorTypeButtonClick, this);
|
|
m_hsvButton.Click.connect(&PaletteEntryEditor::onColorTypeButtonClick, this);
|
|
m_moreOptions.Click.connect(&PaletteEntryEditor::onMoreOptionsClick, this);
|
|
m_copyButton.Click.connect(&PaletteEntryEditor::onCopyColorsClick, this);
|
|
m_pasteButton.Click.connect(&PaletteEntryEditor::onPasteColorsClick, this);
|
|
m_loadButton.Click.connect(&PaletteEntryEditor::onLoadPaletteClick, this);
|
|
m_saveButton.Click.connect(&PaletteEntryEditor::onSavePaletteClick, this);
|
|
m_rampButton.Click.connect(&PaletteEntryEditor::onRampClick, this);
|
|
m_quantizeButton.Click.connect(&PaletteEntryEditor::onQuantizeClick, this);
|
|
|
|
m_rgbSliders.ColorChange.connect(&PaletteEntryEditor::onColorSlidersChange, this);
|
|
m_hsvSliders.ColorChange.connect(&PaletteEntryEditor::onColorSlidersChange, this);
|
|
m_hexColorEntry.ColorChange.connect(&PaletteEntryEditor::onColorHexEntryChange, this);
|
|
|
|
selectColorType(Color::RgbType);
|
|
|
|
// We hook fg/bg color changes (by eyedropper mainly) to update the selected entry color
|
|
app_get_colorbar()->FgColorChange.connect(&PaletteEntryEditor::onFgBgColorChange, this);
|
|
app_get_colorbar()->BgColorChange.connect(&PaletteEntryEditor::onFgBgColorChange, this);
|
|
|
|
// We hook the Frame::Close event to save the frame position before closing it.
|
|
this->Close.connect(Bind<void>(&PaletteEntryEditor::onCloseFrame, this));
|
|
|
|
// We hook App::Exit signal to destroy the g_frame singleton at exit.
|
|
App::instance()->Exit.connect(&PaletteEntryEditor::onExit, this);
|
|
|
|
// Hook for palette change to redraw the palette editor frame
|
|
m_palChangeSlot =
|
|
App::instance()->PaletteChange.connect(&PaletteEntryEditor::onPalChange, this);
|
|
|
|
initTheme();
|
|
}
|
|
|
|
PaletteEntryEditor::~PaletteEntryEditor()
|
|
{
|
|
App::instance()->PaletteChange.disconnect(m_palChangeSlot);
|
|
}
|
|
|
|
void PaletteEntryEditor::setColor(const Color& color)
|
|
{
|
|
m_rgbSliders.setColor(color);
|
|
m_hsvSliders.setColor(color);
|
|
if (!m_disableHexUpdate)
|
|
m_hexColorEntry.setColor(color);
|
|
|
|
PaletteView* palette_editor = app_get_colorbar()->getPaletteView();
|
|
PaletteView::SelectedEntries entries;
|
|
palette_editor->getSelectedEntries(entries);
|
|
int i, j, i2;
|
|
|
|
// Find the first selected entry
|
|
for (i=0; i<(int)entries.size(); ++i)
|
|
if (entries[i])
|
|
break;
|
|
|
|
// Find the first unselected entry after i
|
|
for (i2=i+1; i2<(int)entries.size(); ++i2)
|
|
if (!entries[i2])
|
|
break;
|
|
|
|
// Find the last selected entry
|
|
for (j=entries.size()-1; j>=0; --j)
|
|
if (entries[j])
|
|
break;
|
|
|
|
if (i == j) {
|
|
m_entryLabel.setTextf(" Entry: %d", i);
|
|
}
|
|
else if (j-i+1 == i2-i) {
|
|
m_entryLabel.setTextf(" Range: %d-%d", i, j);
|
|
}
|
|
else if (i == entries.size()) {
|
|
m_entryLabel.setText(" No Entry");
|
|
}
|
|
else {
|
|
m_entryLabel.setText(" Multiple Entries");
|
|
}
|
|
|
|
m_topBox.layout();
|
|
}
|
|
|
|
bool PaletteEntryEditor::onProcessMessage(Message* msg)
|
|
{
|
|
if (msg->type == JM_TIMER &&
|
|
msg->timer.timer == &m_redrawTimer) {
|
|
// Redraw all editors
|
|
if (m_redrawAll) {
|
|
m_redrawAll = false;
|
|
m_implantChange = false;
|
|
m_redrawTimer.stop();
|
|
|
|
// Call all listener of PaletteChange event.
|
|
m_selfPalChange = true;
|
|
App::instance()->PaletteChange();
|
|
m_selfPalChange = false;
|
|
|
|
// Redraw all editors
|
|
try {
|
|
const ActiveDocumentReader document(UIContext::instance());
|
|
update_editors_with_document(document);
|
|
}
|
|
catch (...) {
|
|
// Do nothing
|
|
}
|
|
}
|
|
// Redraw just the current editor
|
|
else {
|
|
m_redrawAll = true;
|
|
current_editor->updateEditor();
|
|
}
|
|
}
|
|
return Frame::onProcessMessage(msg);
|
|
}
|
|
|
|
void PaletteEntryEditor::onExit()
|
|
{
|
|
delete this;
|
|
}
|
|
|
|
void PaletteEntryEditor::onCloseFrame()
|
|
{
|
|
// Save window configuration
|
|
save_window_pos(this, "PaletteEditor");
|
|
|
|
// Uncheck the "Edit Palette" button.
|
|
app_get_colorbar()->setPaletteEditorButtonState(false);
|
|
}
|
|
|
|
void PaletteEntryEditor::onFgBgColorChange(const Color& color)
|
|
{
|
|
if (color.isValid() && color.getType() == Color::IndexType) {
|
|
setColor(color);
|
|
}
|
|
}
|
|
|
|
void PaletteEntryEditor::onColorSlidersChange(ColorSlidersChangeEvent& ev)
|
|
{
|
|
setColor(ev.getColor());
|
|
setPaletteEntryChannel(ev.getColor(), ev.getModifiedChannel());
|
|
updateCurrentSpritePalette("Color Change");
|
|
updateColorBar();
|
|
}
|
|
|
|
void PaletteEntryEditor::onColorHexEntryChange(const Color& color)
|
|
{
|
|
// Disable updating the hex entry so we don't override what the user
|
|
// is writting in the text field.
|
|
m_disableHexUpdate = true;
|
|
|
|
setColor(color);
|
|
setPaletteEntry(color);
|
|
updateCurrentSpritePalette("Color Change");
|
|
updateColorBar();
|
|
|
|
m_disableHexUpdate = false;
|
|
}
|
|
|
|
void PaletteEntryEditor::onColorTypeButtonClick(Event& ev)
|
|
{
|
|
RadioButton* source = static_cast<RadioButton*>(ev.getSource());
|
|
|
|
if (source == &m_rgbButton) selectColorType(Color::RgbType);
|
|
else if (source == &m_hsvButton) selectColorType(Color::HsvType);
|
|
}
|
|
|
|
void PaletteEntryEditor::onMoreOptionsClick(Event& ev)
|
|
{
|
|
Size reqSize;
|
|
|
|
if (m_bottomBox.isVisible()) {
|
|
set_config_bool("PaletteEditor", "ShowMoreOptions", false);
|
|
m_bottomBox.setVisible(false);
|
|
|
|
// Get the required size of the "More options" panel
|
|
reqSize = m_bottomBox.getPreferredSize();
|
|
reqSize.h += 4;
|
|
|
|
// Remove the space occupied by the "More options" panel
|
|
{
|
|
JRect rect = jrect_new(rc->x1, rc->y1,
|
|
rc->x2, rc->y2 - reqSize.h);
|
|
move_window(rect);
|
|
jrect_free(rect);
|
|
}
|
|
}
|
|
else {
|
|
set_config_bool("PaletteEditor", "ShowMoreOptions", true);
|
|
m_bottomBox.setVisible(true);
|
|
|
|
// Get the required size of the whole window
|
|
reqSize = getPreferredSize();
|
|
|
|
// Add space for the "more_options" panel
|
|
if (jrect_h(rc) < reqSize.h) {
|
|
JRect rect = jrect_new(rc->x1, rc->y1,
|
|
rc->x2, rc->y1 + reqSize.h);
|
|
|
|
// Show the expanded area inside the screen
|
|
if (rect->y2 > JI_SCREEN_H)
|
|
jrect_displace(rect, 0, JI_SCREEN_H - rect->y2);
|
|
|
|
move_window(rect);
|
|
jrect_free(rect);
|
|
}
|
|
else
|
|
setBounds(getBounds()); // TODO layout() method is missing
|
|
}
|
|
|
|
// Redraw the window
|
|
invalidate();
|
|
}
|
|
|
|
void PaletteEntryEditor::onCopyColorsClick(Event& ev)
|
|
{
|
|
// Get the selected entries in the palette view.
|
|
PaletteView* palette_editor = app_get_colorbar()->getPaletteView();
|
|
PaletteView::SelectedEntries selectedEntries;
|
|
palette_editor->getSelectedEntries(selectedEntries);
|
|
|
|
// Copy all selected entries into "m_clipboardColors" vector. These
|
|
// entries then are copied back to the palette in the "paste"
|
|
// operation (onPasteColorsClick).
|
|
Palette* palette = get_current_palette();
|
|
m_clipboardColors.clear();
|
|
for (int i=0; i<(int)selectedEntries.size(); ++i)
|
|
if (selectedEntries[i])
|
|
m_clipboardColors.push_back(palette->getEntry(i));
|
|
}
|
|
|
|
void PaletteEntryEditor::onPasteColorsClick(Event& ev)
|
|
{
|
|
// Get the selected entries in the palette view.
|
|
PaletteView* palette_editor = app_get_colorbar()->getPaletteView();
|
|
PaletteView::SelectedEntries selectedEntries;
|
|
palette_editor->getSelectedEntries(selectedEntries);
|
|
|
|
// Count how many colors are selected so we can continue pasting
|
|
// colors even if the current number of selected colors is less than
|
|
// the number of colors into the clipboard.
|
|
int selectedEntriesCount = 0;
|
|
for (int i=0; i<(int)selectedEntries.size(); ++i)
|
|
if (selectedEntries[i])
|
|
selectedEntriesCount++;
|
|
|
|
Palette* palette = get_current_palette();
|
|
for (int i=0, j=0; i<(int)selectedEntries.size() && j<(int)m_clipboardColors.size(); ++i) {
|
|
// The color is pasted if the entry is selected or if the
|
|
// clipboard contains more entries than the current number of
|
|
// selected palette entries.
|
|
if (selectedEntries[i] || j >= selectedEntriesCount)
|
|
palette->setEntry(i, m_clipboardColors[j++]);
|
|
}
|
|
|
|
updateCurrentSpritePalette("Paste Colors");
|
|
updateColorBar();
|
|
|
|
// Generate onPalChange() event so we update all sliders to the new
|
|
// values.
|
|
onPalChange();
|
|
}
|
|
|
|
void PaletteEntryEditor::onLoadPaletteClick(Event& ev)
|
|
{
|
|
Palette *palette;
|
|
base::string filename = app::show_file_selector("Load Palette", "", "png,pcx,bmp,tga,lbm,col,gpl");
|
|
if (!filename.empty()) {
|
|
palette = Palette::load(filename.c_str());
|
|
if (!palette) {
|
|
Alert::show("Error<<Loading palette file||&Close");
|
|
}
|
|
else {
|
|
setNewPalette(palette, "Load Palette");
|
|
delete palette;
|
|
}
|
|
}
|
|
}
|
|
|
|
void PaletteEntryEditor::onSavePaletteClick(Event& ev)
|
|
{
|
|
base::string filename;
|
|
int ret;
|
|
|
|
again:
|
|
filename = app::show_file_selector("Save Palette", "", "png,pcx,bmp,tga,col,gpl");
|
|
if (!filename.empty()) {
|
|
if (exists(filename.c_str())) {
|
|
ret = Alert::show("Warning<<File exists, overwrite it?<<%s||&Yes||&No||&Cancel",
|
|
get_filename(filename.c_str()));
|
|
|
|
if (ret == 2)
|
|
goto again;
|
|
else if (ret != 1)
|
|
return;
|
|
}
|
|
|
|
Palette* palette = get_current_palette();
|
|
if (!palette->save(filename.c_str())) {
|
|
Alert::show("Error<<Saving palette file||&Close");
|
|
}
|
|
}
|
|
}
|
|
|
|
void PaletteEntryEditor::onRampClick(Event& ev)
|
|
{
|
|
PaletteView* palette_editor = app_get_colorbar()->getPaletteView();
|
|
int index1, index2;
|
|
|
|
if (!palette_editor->getSelectedRange(index1, index2))
|
|
return;
|
|
|
|
Palette* src_palette = get_current_palette();
|
|
Palette* dst_palette = new Palette(0, 256);
|
|
|
|
src_palette->copyColorsTo(dst_palette);
|
|
dst_palette->makeHorzRamp(index1, index2);
|
|
|
|
setNewPalette(dst_palette, "Color Ramp");
|
|
delete dst_palette;
|
|
}
|
|
|
|
void PaletteEntryEditor::onQuantizeClick(Event& ev)
|
|
{
|
|
Palette* palette = NULL;
|
|
|
|
{
|
|
const ActiveDocumentReader document(UIContext::instance());
|
|
const Sprite* sprite(document ? document->getSprite(): NULL);
|
|
|
|
if (sprite == NULL) {
|
|
Alert::show("Error<<There is no sprite selected to quantize.||&OK");
|
|
return;
|
|
}
|
|
|
|
if (sprite->getPixelFormat() != IMAGE_RGB) {
|
|
Alert::show("Error<<You can use this command only for RGB sprites||&OK");
|
|
return;
|
|
}
|
|
|
|
palette = quantization::create_palette_from_rgb(sprite);
|
|
}
|
|
|
|
setNewPalette(palette, "Quantize Palette");
|
|
delete palette;
|
|
}
|
|
|
|
void PaletteEntryEditor::setPaletteEntry(const Color& color)
|
|
{
|
|
PaletteView* palView = app_get_colorbar()->getPaletteView();
|
|
PaletteView::SelectedEntries entries;
|
|
palView->getSelectedEntries(entries);
|
|
|
|
uint32_t new_pal_color = _rgba(color.getRed(),
|
|
color.getGreen(),
|
|
color.getBlue(), 255);
|
|
|
|
Palette* palette = get_current_palette();
|
|
for (int c=0; c<palette->size(); c++) {
|
|
if (entries[c])
|
|
palette->setEntry(c, new_pal_color);
|
|
}
|
|
}
|
|
|
|
void PaletteEntryEditor::setPaletteEntryChannel(const Color& color, ColorSliders::Channel channel)
|
|
{
|
|
PaletteView* palView = app_get_colorbar()->getPaletteView();
|
|
PaletteView::SelectedEntries entries;
|
|
palView->getSelectedEntries(entries);
|
|
|
|
int begSel, endSel;
|
|
if (!palView->getSelectedRange(begSel, endSel))
|
|
return;
|
|
|
|
uint32_t src_color;
|
|
int r, g, b;
|
|
|
|
Palette* palette = get_current_palette();
|
|
for (int c=0; c<palette->size(); c++) {
|
|
if (entries[c]) {
|
|
// Get the current RGB values of the palette entry
|
|
src_color = palette->getEntry(c);
|
|
r = _rgba_getr(src_color);
|
|
g = _rgba_getg(src_color);
|
|
b = _rgba_getb(src_color);
|
|
|
|
switch (color.getType()) {
|
|
|
|
case Color::RgbType:
|
|
// Modify one entry
|
|
if (begSel == endSel) {
|
|
r = color.getRed();
|
|
g = color.getGreen();
|
|
b = color.getBlue();
|
|
}
|
|
// Modify one channel a set of entries
|
|
else {
|
|
// Setup the new RGB values depending of the modified channel.
|
|
switch (channel) {
|
|
case ColorSliders::Red:
|
|
r = color.getRed();
|
|
case ColorSliders::Green:
|
|
g = color.getGreen();
|
|
break;
|
|
case ColorSliders::Blue:
|
|
b = color.getBlue();
|
|
break;
|
|
}
|
|
}
|
|
break;
|
|
|
|
case Color::HsvType:
|
|
{
|
|
Hsv hsv;
|
|
|
|
// Modify one entry
|
|
if (begSel == endSel) {
|
|
hsv.hue(color.getHue());
|
|
hsv.saturation(double(color.getSaturation()) / 100.0);
|
|
hsv.value(double(color.getValue()) / 100.0);
|
|
}
|
|
// Modify one channel a set of entries
|
|
else {
|
|
// Convert RGB to HSV
|
|
hsv = Hsv(Rgb(r, g, b));
|
|
|
|
// Only modify the desired HSV channel
|
|
switch (channel) {
|
|
case ColorSliders::Hue:
|
|
hsv.hue(color.getHue());
|
|
break;
|
|
case ColorSliders::Saturation:
|
|
hsv.saturation(double(color.getSaturation()) / 100.0);
|
|
break;
|
|
case ColorSliders::Value:
|
|
hsv.value(double(color.getValue()) / 100.0);
|
|
break;
|
|
}
|
|
}
|
|
|
|
// Convert HSV back to RGB
|
|
Rgb rgb(hsv);
|
|
r = rgb.red();
|
|
g = rgb.green();
|
|
b = rgb.blue();
|
|
}
|
|
break;
|
|
}
|
|
|
|
palette->setEntry(c, _rgba(r, g, b, 255));
|
|
}
|
|
}
|
|
}
|
|
|
|
void PaletteEntryEditor::selectColorType(Color::Type type)
|
|
{
|
|
m_rgbSliders.setVisible(type == Color::RgbType);
|
|
m_hsvSliders.setVisible(type == Color::HsvType);
|
|
|
|
switch (type) {
|
|
case Color::RgbType: m_rgbButton.setSelected(true); break;
|
|
case Color::HsvType: m_hsvButton.setSelected(true); break;
|
|
}
|
|
|
|
m_vbox.layout();
|
|
m_vbox.invalidate();
|
|
}
|
|
|
|
void PaletteEntryEditor::setNewPalette(Palette* palette, const char* operationName)
|
|
{
|
|
// Copy the palette
|
|
palette->copyColorsTo(get_current_palette());
|
|
|
|
// Set the palette calling the hooks
|
|
set_current_palette(palette, false);
|
|
|
|
// Update the sprite palette
|
|
updateCurrentSpritePalette(operationName);
|
|
|
|
// Redraw the entire screen
|
|
ui::Manager::getDefault()->invalidate();
|
|
}
|
|
|
|
void PaletteEntryEditor::updateCurrentSpritePalette(const char* operationName)
|
|
{
|
|
if (UIContext::instance()->getActiveDocument() &&
|
|
UIContext::instance()->getActiveDocument()->getSprite()) {
|
|
try {
|
|
ActiveDocumentWriter document(UIContext::instance());
|
|
Sprite* sprite(document->getSprite());
|
|
Palette* newPalette = get_current_palette(); // System current pal
|
|
Palette* currentSpritePalette = sprite->getPalette(sprite->getCurrentFrame()); // Sprite current pal
|
|
int from, to;
|
|
|
|
// Check differences between current sprite palette and current system palette
|
|
from = to = -1;
|
|
currentSpritePalette->countDiff(newPalette, &from, &to);
|
|
|
|
if (from >= 0 && to >= from) {
|
|
DocumentUndo* undo = document->getUndo();
|
|
|
|
// Add undo information to save the range of pal entries that will be modified.
|
|
if (undo->isEnabled()) {
|
|
if (m_implantChange && strcmp(undo->getNextUndoLabel(), operationName) == 0) {
|
|
undo->implantUndoerInLastGroup
|
|
(new undoers::SetPaletteColors(undo->getObjects(),
|
|
sprite, currentSpritePalette, from, to));
|
|
}
|
|
else {
|
|
UndoTransaction undoTransaction(document, operationName, undo::ModifyDocument);
|
|
undoTransaction.pushUndoer
|
|
(new undoers::SetPaletteColors(undoTransaction.getObjects(),
|
|
sprite, currentSpritePalette, from, to));
|
|
undoTransaction.commit();
|
|
}
|
|
}
|
|
|
|
// Change the sprite palette
|
|
sprite->setPalette(newPalette, false);
|
|
}
|
|
}
|
|
catch (base::Exception& e) {
|
|
Console::showException(e);
|
|
}
|
|
}
|
|
|
|
PaletteView* palette_editor = app_get_colorbar()->getPaletteView();
|
|
palette_editor->invalidate();
|
|
|
|
if (!m_redrawTimer.isRunning())
|
|
m_redrawTimer.start();
|
|
|
|
m_redrawAll = false;
|
|
m_implantChange = true;
|
|
}
|
|
|
|
void PaletteEntryEditor::updateColorBar()
|
|
{
|
|
app_get_colorbar()->invalidate();
|
|
}
|
|
|
|
void PaletteEntryEditor::onPalChange()
|
|
{
|
|
if (!m_selfPalChange) {
|
|
PaletteView* palette_editor = app_get_colorbar()->getPaletteView();
|
|
int index = palette_editor->getSelectedEntry();
|
|
if (index >= 0)
|
|
setColor(Color::fromIndex(index));
|
|
|
|
// Redraw the window
|
|
invalidate();
|
|
}
|
|
}
|
|
|
|
//////////////////////////////////////////////////////////////////////
|
|
// CommandFactory
|
|
|
|
Command* CommandFactory::createPaletteEditorCommand()
|
|
{
|
|
return new PaletteEditorCommand;
|
|
}
|