Make timeline and new text rendering appear on premium only and remove license registration

pull/300/head
James H Ball 2025-04-25 18:14:27 +01:00
rodzic ab2fcc287f
commit db8206200b
14 zmienionych plików z 20 dodań i 330 usunięć

Wyświetl plik

@ -35,7 +35,7 @@ CommonPluginEditor::CommonPluginEditor(CommonAudioProcessor& p, juce::String app
}
}
addChildComponent(visualiser);
addAndMakeVisible(visualiser);
int width = std::any_cast<int>(audioProcessor.getProperty("appWidth", defaultWidth));
int height = std::any_cast<int>(audioProcessor.getProperty("appHeight", defaultHeight));
@ -66,8 +66,6 @@ CommonPluginEditor::CommonPluginEditor(CommonAudioProcessor& p, juce::String app
#if OSCI_PREMIUM
sharedTextureManager.initGL();
#else
visualiser.setVisible(true);
#endif
}

Wyświetl plik

@ -10,10 +10,6 @@
#include "components/VolumeComponent.h"
#include "components/DownloaderComponent.h"
#if OSCI_PREMIUM
#include "components/LicenseRegistrationComponent.h"
#endif
class CommonPluginEditor : public juce::AudioProcessorEditor {
public:
CommonPluginEditor(CommonAudioProcessor&, juce::String appName, juce::String projectFileType, int width, int height);
@ -76,11 +72,6 @@ public:
juce::SharedResourcePointer<juce::TooltipWindow> tooltipWindow;
juce::DropShadower tooltipDropShadow{juce::DropShadow(juce::Colours::black.withAlpha(0.5f), 6, {0,0})};
#if OSCI_PREMIUM
LicenseRegistrationComponent licenseRegistration {audioProcessor, [this](bool success) {
visualiser.setVisible(success);
}};
#endif
bool usingNativeMenuBar = false;
#if JUCE_LINUX

Wyświetl plik

@ -89,8 +89,6 @@ public:
bool hasSetSessionStartTime = false;
bool programCrashedAndUserWantsToReset();
std::atomic<bool> licenseVerified = true;
juce::SpinLock audioPlayerListenersLock;
std::vector<AudioPlayerListener*> audioPlayerListeners;

Wyświetl plik

@ -16,7 +16,9 @@ FrameSettingsComponent::FrameSettingsComponent(OscirenderAudioProcessor& p, Osci
offsetLabel.setTooltip("Offsets the animation's start point by a specified number of frames.");
} else {
audioProcessor.animationSyncBPM->setValueNotifyingHost(false);
#if OSCI_PREMIUM
addAndMakeVisible(timeline);
#endif
}
addAndMakeVisible(rateLabel);
addAndMakeVisible(rateBox);
@ -60,20 +62,26 @@ void FrameSettingsComponent::resized() {
auto area = getLocalBounds().withTrimmedTop(20).reduced(20);
double rowHeight = 20;
#if OSCI_PREMIUM
auto timelineArea = juce::JUCEApplicationBase::isStandaloneApp() ? area.removeFromBottom(30) : juce::Rectangle<int>();
#endif
auto toggleBounds = juce::JUCEApplicationBase::isStandaloneApp() ? juce::Rectangle<int>() : area.removeFromTop(rowHeight);
auto toggleWidth = juce::jmin(area.getWidth() / 3, 150);
auto firstColumn = area.removeFromLeft(220);
#if OSCI_PREMIUM
if (juce::JUCEApplicationBase::isStandaloneApp()) {
timeline.setVisible(animated);
}
#endif
if (animated) {
if (juce::JUCEApplicationBase::isStandaloneApp()) {
#if OSCI_PREMIUM
timeline.setBounds(timelineArea);
#endif
} else {
animate.setBounds(toggleBounds.removeFromLeft(toggleWidth));
sync.setBounds(toggleBounds.removeFromLeft(toggleWidth));

Wyświetl plik

@ -33,7 +33,9 @@ private:
juce::Label offsetLabel{ "Offset","Offset" };
DoubleTextBox rateBox{ audioProcessor.animationRate->min, audioProcessor.animationRate->max };
DoubleTextBox offsetBox{ audioProcessor.animationOffset->min, audioProcessor.animationRate->max };
#if OSCI_PREMIUM
AnimationTimelineComponent timeline{audioProcessor};
#endif
jux::SwitchButton invertImage{audioProcessor.invertImage};
EffectComponent threshold{*audioProcessor.imageThreshold};

Wyświetl plik

@ -128,7 +128,7 @@ MainComponent::MainComponent(OscirenderAudioProcessor& p, OscirenderAudioProcess
osci::BooleanParameter* visualiserFullScreen = audioProcessor.visualiserParameters.visualiserFullScreen;
pluginEditor.visualiser.setFullScreen(visualiserFullScreen->getBoolValue());
addChildComponent(pluginEditor.visualiser);
addAndMakeVisible(pluginEditor.visualiser);
pluginEditor.visualiser.setFullScreenCallback([this, visualiserFullScreen](FullScreenMode mode) {
if (mode == FullScreenMode::TOGGLE) {
visualiserFullScreen->setBoolValueNotifyingHost(!visualiserFullScreen->getBoolValue());

Wyświetl plik

@ -77,7 +77,7 @@ OscirenderAudioProcessorEditor::OscirenderAudioProcessorEditor(OscirenderAudioPr
addAndMakeVisible(lua);
addAndMakeVisible(luaResizerBar);
addChildComponent(visualiser);
addAndMakeVisible(visualiser);
visualiser.openSettings = [this] {
openVisualiserSettings();
@ -95,11 +95,6 @@ OscirenderAudioProcessorEditor::OscirenderAudioProcessorEditor(OscirenderAudioPr
#endif
initialiseMenuBar(model);
#if OSCI_PREMIUM
addChildComponent(licenseRegistration);
licenseRegistration.toFront(true);
#endif
}
OscirenderAudioProcessorEditor::~OscirenderAudioProcessorEditor() {
@ -189,10 +184,6 @@ void OscirenderAudioProcessorEditor::resized() {
auto area = getLocalBounds();
#if OSCI_PREMIUM
licenseRegistration.setBounds(area);
#endif
if (audioProcessor.visualiserParameters.visualiserFullScreen->getBoolValue()) {
visualiser.setBounds(area);
return;

Wyświetl plik

@ -620,7 +620,7 @@ void OscirenderAudioProcessor::processBlock(juce::AudioBuffer<float>& buffer, ju
threadManager.write(osci::Point(x, y, 1));
// Apply mute if active
if (muteParameter->getBoolValue() || !licenseVerified) {
if (muteParameter->getBoolValue()) {
x = 0.0;
y = 0.0;
}

Wyświetl plik

@ -33,11 +33,6 @@ SosciPluginEditor::SosciPluginEditor(SosciAudioProcessor& p) : CommonPluginEdito
manager.addChangeListener(this);
currentInputDevice = getInputDeviceName();
}
#if OSCI_PREMIUM
addChildComponent(licenseRegistration);
licenseRegistration.toFront(true);
#endif
}
SosciPluginEditor::~SosciPluginEditor() {
@ -58,10 +53,6 @@ void SosciPluginEditor::resized() {
CommonPluginEditor::resized();
auto area = getLocalBounds();
#if OSCI_PREMIUM
licenseRegistration.setBounds(area);
#endif
if (audioProcessor.visualiserParameters.visualiserFullScreen->getBoolValue()) {
visualiser.setBounds(area);
} else {

Wyświetl plik

@ -1,246 +0,0 @@
#include "LicenseRegistrationComponent.h"
LicenseRegistrationComponent::LicenseRegistrationComponent(CommonAudioProcessor& processor, std::function<void(bool)> onLicenseVerified)
: audioProcessor(processor), onLicenseVerified(onLicenseVerified)
{
setupComponents();
auto showComponent = [this] {
// If validated within the last week, show immediately
juce::WeakReference<LicenseRegistrationComponent> weakThis(this);
juce::MessageManager::callAsync([weakThis]() {
if (auto* strongThis = weakThis.get()) {
strongThis->setVisible(true);
}
});
audioProcessor.licenseVerified = false;
};
audioProcessor.reloadGlobalSettings();
auto savedKey = audioProcessor.getGlobalStringValue("license_key");
if (savedKey.isNotEmpty())
{
// Pre-populate the license key field
licenseKeyEditor.setText(savedKey, false);
auto lastValidated = audioProcessor.getGlobalStringValue("license_last_validated");
if (lastValidated.isNotEmpty())
{
auto lastValidationTime = juce::Time::fromISO8601(lastValidated);
auto weekAgo = juce::Time::getCurrentTime() - juce::RelativeTime::weeks(1);
auto hourAgo = juce::Time::getCurrentTime() - juce::RelativeTime::hours(1);
if (lastValidationTime > weekAgo)
{
if (onLicenseVerified != nullptr) {
onLicenseVerified(true);
}
audioProcessor.licenseVerified = true;
} else {
showComponent();
}
if (lastValidationTime < hourAgo) {
// Validate the license key in the background
validateSavedLicense();
}
// Start periodic checks every hour
startTimer(1000 * 60 * 60);
}
} else {
showComponent();
}
}
LicenseRegistrationComponent::~LicenseRegistrationComponent()
{
stopTimer();
}
void LicenseRegistrationComponent::setupComponents()
{
titleLabel.setText("License Registration Required", juce::dontSendNotification);
titleLabel.setFont(juce::Font(24.0f, juce::Font::bold));
titleLabel.setJustificationType(juce::Justification::centred);
addAndMakeVisible(titleLabel);
instructionsLabel.setText("Please enter your license key to continue", juce::dontSendNotification);
instructionsLabel.setJustificationType(juce::Justification::centred);
instructionsLabel.setFont(juce::Font(18.0f));
addAndMakeVisible(instructionsLabel);
licenseKeyEditor.setTextToShowWhenEmpty("XXXXXXXX-XXXXXXXX-XXXXXXXX-XXXXXXXX", juce::Colours::grey);
licenseKeyEditor.setInputRestrictions(35, "0123456789ABCDEFabcdef-"); // Allow both upper and lowercase hex digits and hyphens
licenseKeyEditor.onReturnKey = [this] { verifyButton.triggerClick(); };
licenseKeyEditor.setFont(juce::Font(24.0f));
licenseKeyEditor.onTextChange = [this] {
auto currentText = licenseKeyEditor.getText();
auto upperText = currentText.toUpperCase();
if (currentText != upperText)
{
auto cursorPos = licenseKeyEditor.getCaretPosition();
licenseKeyEditor.setText(upperText, false);
licenseKeyEditor.setCaretPosition(cursorPos);
}
};
addAndMakeVisible(licenseKeyEditor);
verifyButton.setButtonText("Verify License");
verifyButton.onClick = [this] {
if (!isVerifying)
{
verifyLicense(licenseKeyEditor.getText());
}
};
addAndMakeVisible(verifyButton);
}
void LicenseRegistrationComponent::paint(juce::Graphics& g)
{
g.fillAll(juce::Colours::black.withAlpha(0.9f));
auto bounds = getLocalBounds().toFloat();
}
void LicenseRegistrationComponent::resized()
{
auto bounds = getLocalBounds().reduced(20);
titleLabel.setBounds(bounds.removeFromTop(40));
bounds.removeFromTop(20);
instructionsLabel.setBounds(bounds.removeFromTop(30));
bounds.removeFromTop(20);
auto row = bounds.removeFromTop(35);
licenseKeyEditor.setBounds(row.reduced(50, 0));
bounds.removeFromTop(20);
auto buttonBounds = bounds.removeFromTop(40);
verifyButton.setBounds(buttonBounds.withSizeKeepingCentre(120, 40));
}
void LicenseRegistrationComponent::verifyLicense(const juce::String& licenseKey, bool showErrorDialog)
{
if (licenseKey.isEmpty())
return;
isVerifying = true;
verifyButton.setEnabled(false);
verifyButton.setVisible(false);
juce::URL url("https://api.osci-render.com/api/verify-license");
auto jsonObj = std::make_unique<juce::DynamicObject>();
jsonObj->setProperty("license_key", licenseKey);
juce::var jsonData(jsonObj.release());
url = url.withPOSTData(juce::JSON::toString(jsonData));
auto webStream = url.createInputStream(false, nullptr, nullptr,
"Content-Type: application/json",
10000);
bool successfullyVerified = false;
if (webStream != nullptr)
{
auto response = webStream->readEntireStreamAsString();
DBG(response);
auto json = juce::JSON::parse(response);
if (json.hasProperty("success")) {
bool success = json["success"];
if (success && json.hasProperty("valid") && json.hasProperty("purchase")) {
bool valid = json["valid"];
auto purchase = json["purchase"].getDynamicObject();
auto productId = purchase->getProperty("product_id").toString();
if (success && valid && productId == SOSCI_PRODUCT_ID)
{
// Save the license key and validation timestamp
audioProcessor.setGlobalValue("license_key", licenseKey);
audioProcessor.setGlobalValue("license_last_validated", juce::Time::getCurrentTime().toISO8601(true));
audioProcessor.saveGlobalSettings();
audioProcessor.licenseVerified = true;
successfullyVerified = true;
juce::WeakReference<LicenseRegistrationComponent> weakThis(this);
juce::MessageManager::callAsync([weakThis]() {
if (auto* strongThis = weakThis.get()) {
strongThis->setVisible(false);
}
});
startTimer(1000 * 60 * 60); // Check every hour
}
else if (showErrorDialog)
{
juce::AlertWindow::showMessageBoxAsync(juce::AlertWindow::WarningIcon,
"Invalid License",
"The license key you entered is not valid. Please check and try again.");
}
else
{
// Background check failed, clear the license
clearLicense();
}
} else if (showErrorDialog && json.hasProperty("message")) {
auto message = json["message"].toString();
juce::AlertWindow::showMessageBoxAsync(juce::AlertWindow::WarningIcon, "Error", message);
}
}
}
else if (showErrorDialog)
{
juce::AlertWindow::showMessageBoxAsync(juce::AlertWindow::WarningIcon,
"Connection Error",
"Could not connect to the license server. Please check your internet connection and try again.");
}
isVerifying = false;
verifyButton.setEnabled(true);
verifyButton.setVisible(true);
if (onLicenseVerified != nullptr) {
onLicenseVerified(successfullyVerified);
}
}
bool LicenseRegistrationComponent::validateSavedLicense()
{
auto savedKey = audioProcessor.getGlobalStringValue("license_key");
if (savedKey.isNotEmpty())
{
auto lastValidated = audioProcessor.getGlobalStringValue("license_last_validated");
if (lastValidated.isNotEmpty())
{
// Verify in the background without showing error dialogs
verifyLicense(savedKey, false);
return true;
}
}
return false;
}
void LicenseRegistrationComponent::clearLicense()
{
audioProcessor.removeGlobalValue("license_key");
audioProcessor.removeGlobalValue("license_last_validated");
audioProcessor.saveGlobalSettings();
setVisible(true);
audioProcessor.licenseVerified = false;
}
void LicenseRegistrationComponent::timerCallback()
{
auto savedKey = audioProcessor.getGlobalStringValue("license_key");
if (savedKey.isNotEmpty()) {
verifyLicense(savedKey, false);
}
}

Wyświetl plik

@ -1,38 +0,0 @@
#pragma once
#include <JuceHeader.h>
#include "../CommonPluginProcessor.h"
class LicenseRegistrationComponent : public juce::Component,
public juce::Timer
{
public:
LicenseRegistrationComponent(CommonAudioProcessor& processor, std::function<void(bool)> onLicenseVerified);
~LicenseRegistrationComponent() override;
void paint(juce::Graphics& g) override;
void resized() override;
// Timer callback for background license verification
void timerCallback() override;
private:
void verifyLicense(const juce::String& licenseKey, bool showErrorDialog = true);
bool validateSavedLicense();
void clearLicense();
void setupComponents();
CommonAudioProcessor& audioProcessor;
juce::Label titleLabel;
juce::Label instructionsLabel;
juce::TextEditor licenseKeyEditor;
juce::TextButton verifyButton;
bool isVerifying = false;
const juce::String SOSCI_PRODUCT_ID = "Hsr9C58_YhTxYP0MNvsIow==";
std::function<void(bool)> onLicenseVerified;
JUCE_DECLARE_NON_COPYABLE_WITH_LEAK_DETECTOR(LicenseRegistrationComponent)
JUCE_DECLARE_WEAK_REFERENCEABLE(LicenseRegistrationComponent)
};

Wyświetl plik

@ -13,6 +13,9 @@ TextParser::~TextParser() {
void TextParser::parse(juce::String text, juce::Font font) {
lastFont = font;
juce::Path textPath;
#if OSCI_PREMIUM
// Apply formatting markers if the font is bold or italic
juce::String formattedText = text;
@ -31,9 +34,6 @@ void TextParser::parse(juce::String text, juce::Font font) {
juce::TextLayout layout;
layout.createLayout(attributedString, 64.0f);
// Create a path from the TextLayout
juce::Path textPath;
juce::String displayText = attributedString.getText();
// remove all whitespace
displayText = displayText.removeCharacters(" \t\n\r");
@ -75,10 +75,13 @@ void TextParser::parse(juce::String text, juce::Font font) {
// If the layout has no text, fallback to original method
if (textPath.isEmpty()) {
#endif
juce::GlyphArrangement glyphs;
glyphs.addFittedText(font, text, -2, -2, 4, 4, juce::Justification::centred, 2);
glyphs.createPath(textPath);
#if OSCI_PREMIUM
}
#endif
// Convert path to shapes
shapes = std::vector<std::unique_ptr<osci::Shape>>();

Wyświetl plik

@ -162,10 +162,6 @@
file="Source/components/ErrorCodeEditorComponent.h"/>
<FILE id="L9DIT2" name="LabelledTextBox.h" compile="0" resource="0"
file="Source/components/LabelledTextBox.h"/>
<FILE id="mp7cFK" name="LicenseRegistrationComponent.cpp" compile="1"
resource="0" file="Source/components/LicenseRegistrationComponent.cpp"/>
<FILE id="eW7t3h" name="LicenseRegistrationComponent.h" compile="0"
resource="0" file="Source/components/LicenseRegistrationComponent.h"/>
<FILE id="tpNWJ3" name="LuaConsole.cpp" compile="1" resource="0" file="Source/components/LuaConsole.cpp"/>
<FILE id="e3ZteP" name="LuaConsole.h" compile="0" resource="0" file="Source/components/LuaConsole.h"/>
<FILE id="qIxm1z" name="LuaListComponent.cpp" compile="1" resource="0"

Wyświetl plik

@ -120,10 +120,6 @@
file="Source/components/EffectComponent.cpp"/>
<FILE id="u4UCwb" name="EffectComponent.h" compile="0" resource="0"
file="Source/components/EffectComponent.h"/>
<FILE id="GPZ9q4" name="LicenseRegistrationComponent.cpp" compile="1"
resource="0" file="Source/components/LicenseRegistrationComponent.cpp"/>
<FILE id="JcXyLX" name="LicenseRegistrationComponent.h" compile="0"
resource="0" file="Source/components/LicenseRegistrationComponent.h"/>
<FILE id="ka6rAh" name="MainMenuBarModel.cpp" compile="1" resource="0"
file="Source/components/MainMenuBarModel.cpp"/>
<FILE id="t6oUhv" name="MainMenuBarModel.h" compile="0" resource="0"