1
0
Fork 0
mirror of https://github.com/juce-framework/JUCE.git synced 2026-01-13 00:04:19 +00:00
JUCE/extras/Projucer/Source/Project/jucer_DependencyPathPropertyComponent.h

270 lines
9.8 KiB
C++

/*
==============================================================================
This file is part of the JUCE library.
Copyright (c) 2017 - ROLI Ltd.
JUCE is an open source library subject to commercial or open-source
licensing.
By using JUCE, you agree to the terms of both the JUCE 5 End-User License
Agreement and JUCE 5 Privacy Policy (both updated and effective as of the
27th April 2017).
End User License Agreement: www.juce.com/juce-5-licence
Privacy Policy: www.juce.com/juce-5-privacy-policy
Or: You may also use this code under the terms of the GPL v3 (see
www.gnu.org/licenses).
JUCE IS PROVIDED "AS IS" WITHOUT ANY WARRANTY, AND ALL WARRANTIES, WHETHER
EXPRESSED OR IMPLIED, INCLUDING MERCHANTABILITY AND FITNESS FOR PURPOSE, ARE
DISCLAIMED.
==============================================================================
*/
#pragma once
//==============================================================================
/** This ValueSource type implements the fallback logic required for dependency
path settings: use the project exporter value; if this is empty, fall back to
the global preference value; if the exporter is supposed to run on another
OS and we don't know what the global preferences on that other machine are,
fall back to a generic OS-specific fallback value.
*/
class DependencyPathValueSource : public Value::ValueSource,
private Value::Listener
{
public:
DependencyPathValueSource (const Value& projectSettingsPath,
Identifier globalSettingsKey,
DependencyPathOS osThisSettingAppliesTo);
/** This gets the currently used value, which may be either
the project setting, the global setting, or the fallback value. */
var getValue() const override
{
if (isUsingProjectSettings())
return projectSettingsValue.getValue();
if (isUsingGlobalSettings())
return globalSettingsValue.getValue();
return fallbackValue.getValue();
}
void setValue (const var& newValue) override
{
projectSettingsValue = newValue;
if (isUsingProjectSettings())
sendChangeMessage (false);
}
bool isUsingProjectSettings() const
{
return projectSettingsValueIsValid();
}
bool isUsingGlobalSettings() const
{
return ! projectSettingsValueIsValid() && globalSettingsValueIsValid();
}
bool isUsingFallbackValue() const
{
return ! projectSettingsValueIsValid() && ! globalSettingsValueIsValid();
}
bool appliesToThisOS() const
{
return os == TargetOS::getThisOS();
}
bool isValidPath (const File& relativeTo) const;
bool isValidPath() const;
Identifier getKey() { return globalKey; }
Value getGlobalSettingsValue() { return globalSettingsValue; }
Value getFallbackSettingsValue() { return fallbackValue; }
private:
void valueChanged (Value& value) override
{
if ((value.refersToSameSourceAs (globalSettingsValue) && isUsingGlobalSettings())
|| (value.refersToSameSourceAs (fallbackValue) && isUsingFallbackValue()))
{
sendChangeMessage (true);
setValue (String()); // make sure that the project-specific value is still blank
}
}
/** This defines when to use the project setting, and when to
consider it invalid and to fall back to the global setting or
the fallback value. */
bool projectSettingsValueIsValid() const
{
return ! projectSettingsValue.toString().isEmpty();
}
/** This defines when to use the global setting - given the project setting
is invalid, and when to fall back to the fallback value instead. */
bool globalSettingsValueIsValid() const
{
// only use the global settings if they are set on the same OS
// that this setting is for!
DependencyPathOS thisOS = TargetOS::getThisOS();
return thisOS == TargetOS::unknown ? false : os == thisOS;
}
/** the dependency path setting as set in this Projucer project. */
Value projectSettingsValue;
/** the global key used in the application settings for the global setting value.
needed for checking whether the path is valid. */
Identifier globalKey;
/** on what operating system should this dependency path be used?
note that this is *not* the os that is targeted by the project,
but rather the os on which the project will be compiled
(= on which the path settings need to be set correctly). */
DependencyPathOS os;
/** the dependency path global setting on this machine.
used when there value set for this project is invalid. */
Value globalSettingsValue;
/** the dependency path fallback setting. used instead of the global setting
whenever the latter doesn't apply, e.g. the setting is for another
OS than the ome this machine is running. */
Value fallbackValue;
};
//==============================================================================
class DependencyPathPropertyComponent : public TextPropertyComponent,
private Value::Listener,
private Label::Listener
{
public:
DependencyPathPropertyComponent (const File& pathRelativeToUse,
const Value& value,
const String& propertyName);
private:
/** This function defines what colour the label text should assume
depending on the current state of the value the component tracks. */
Colour getTextColourToDisplay() const;
/** This function handles path changes because of user input. */
void textWasEdited() override;
/** This function handles path changes because the global path changed. */
void valueChanged (Value& value) override;
/** If the dependency path is relative, relative to which directory should
we check if an object is available. */
File pathRelativeTo;
/** the value that represents this dependency path setting. */
Value pathValue;
/** a reference to the value source that this value refers to. */
DependencyPathValueSource& pathValueSource;
// Label::Listener overrides:
void labelTextChanged (Label* labelThatHasChanged) override;
void editorShown (Label*, TextEditor&) override;
void editorHidden (Label*, TextEditor&) override;
void lookAndFeelChanged() override;
JUCE_DECLARE_NON_COPYABLE_WITH_LEAK_DETECTOR (DependencyPathPropertyComponent)
};
//==============================================================================
class DependencyFilePathPropertyComponent : public TextPropertyComponent,
public FileDragAndDropTarget,
private Value::Listener,
private Label::Listener,
private Button::Listener
{
public:
DependencyFilePathPropertyComponent (Value& value,
const String& propertyDescription,
bool isDirectory,
const String& wildcards = "*",
const File& rootToUseForRelativePaths = File());
void resized() override;
void paintOverChildren (Graphics& g) override;
bool isInterestedInFileDrag (const StringArray&) override { return isEnabled(); }
void fileDragEnter (const StringArray&, int, int) override { highlightForDragAndDrop = true; repaint(); }
void fileDragExit (const StringArray&) override { highlightForDragAndDrop = false; repaint(); }
void filesDropped (const StringArray&, int, int) override;
void setTo (const File& f);
void enablementChanged() override;
private:
void textWasEdited() override;
void valueChanged (Value&) override;
void labelTextChanged (Label*) override {}
void editorHidden (Label*, TextEditor&) override {}
void editorShown (Label*, TextEditor&) override;
void buttonClicked (Button*) override;
void lookAndFeelChanged() override
{
browseButton.setColour (TextButton::buttonColourId,
findColour (secondaryButtonBackgroundColourId));
textWasEdited();
}
Colour getTextColourToDisplay() const;
//==========================================================================
File pathRelativeTo;
Value pathValue;
DependencyPathValueSource& pathValueSource;
TextButton browseButton;
bool isDirectory, highlightForDragAndDrop = false;
String wildcards;
JUCE_DECLARE_NON_COPYABLE_WITH_LEAK_DETECTOR (DependencyFilePathPropertyComponent)
};
//==============================================================================
class TextPropertyComponentWithEnablement : public TextPropertyComponent,
private Value::Listener
{
public:
TextPropertyComponentWithEnablement (const Value& valueToControl, const Value& valueToListenTo,
const String& propertyName, int maxNumChars, bool isMultiLine)
: TextPropertyComponent (valueToControl, propertyName, maxNumChars, isMultiLine),
value (valueToListenTo)
{
value.addListener (this);
setEnabled (value.getValue());
}
private:
Value value;
void valueChanged (Value& v) override
{
setEnabled (v.getValue());
}
};