/* ============================================================================== This file is part of the JUCE library. Copyright (c) 2020 - Raw Material Software Limited 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 6 End-User License Agreement and JUCE Privacy Policy (both effective as of the 16th June 2020). End User License Agreement: www.juce.com/juce-6-licence Privacy Policy: www.juce.com/juce-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. ============================================================================== */ namespace juce { extern Image JUCE_API getIconFromApplication (const String&, int); static Image getIconFromIcnsFile (const File& icnsFile, const int size) { FileInputStream stream (icnsFile); if (! stream.openedOk()) return {}; const int numHeaderSectionBytes = 4; char headerSection [numHeaderSectionBytes]; if (stream.read (headerSection, numHeaderSectionBytes) != numHeaderSectionBytes || headerSection[0] != 'i' || headerSection[1] != 'c' || headerSection[2] != 'n' || headerSection[3] != 's') return {}; if (stream.read (headerSection, numHeaderSectionBytes) != numHeaderSectionBytes) return {}; const auto dataSize = juce::ByteOrder::bigEndianInt (headerSection); if (dataSize <= 0) return {}; OwnedArray internalFormats; internalFormats.add (new PNGImageFormat()); internalFormats.add (new JPEGImageFormat()); Array images; auto maxWidth = 0; auto maxWidthIndex = -1; while (stream.getPosition() < dataSize) { const auto sectionStart = stream.getPosition(); if (! stream.setPosition (sectionStart + 4)) break; if (stream.read (headerSection, numHeaderSectionBytes) != numHeaderSectionBytes) break; const auto sectionSize = ByteOrder::bigEndianInt (headerSection); if (sectionSize <= 0) break; const auto sectionDataStart = stream.getPosition(); for (auto* fmt : internalFormats) { if (fmt->canUnderstand (stream)) { stream.setPosition (sectionDataStart); images.add (fmt->decodeImage (stream)); const auto lastImageIndex = images.size() - 1; const auto lastWidth = images.getReference (lastImageIndex).getWidth(); if (lastWidth > maxWidth) { maxWidthIndex = lastImageIndex; maxWidth = lastWidth; } } stream.setPosition (sectionDataStart); } stream.setPosition (sectionStart + sectionSize); } return maxWidthIndex == -1 ? juce::Image() : images.getReference (maxWidthIndex).rescaled (size, size, Graphics::ResamplingQuality::highResamplingQuality); } Image JUCE_API getIconFromApplication (const String& applicationPath, const int size) { Image hostIcon; if (CFStringRef pathCFString = CFStringCreateWithCString (kCFAllocatorDefault, applicationPath.toRawUTF8(), kCFStringEncodingUTF8)) { if (CFURLRef url = CFURLCreateWithFileSystemPath (kCFAllocatorDefault, pathCFString, kCFURLPOSIXPathStyle, 1)) { if (CFBundleRef appBundle = CFBundleCreate (kCFAllocatorDefault, url)) { if (CFTypeRef infoValue = CFBundleGetValueForInfoDictionaryKey (appBundle, CFSTR("CFBundleIconFile"))) { if (CFGetTypeID (infoValue) == CFStringGetTypeID()) { CFStringRef iconFilename = reinterpret_cast (infoValue); CFStringRef resourceURLSuffix = CFStringHasSuffix (iconFilename, CFSTR(".icns")) ? nullptr : CFSTR("icns"); if (CFURLRef iconURL = CFBundleCopyResourceURL (appBundle, iconFilename, resourceURLSuffix, nullptr)) { if (CFStringRef iconPath = CFURLCopyFileSystemPath (iconURL, kCFURLPOSIXPathStyle)) { File icnsFile (CFStringGetCStringPtr (iconPath, CFStringGetSystemEncoding())); hostIcon = getIconFromIcnsFile (icnsFile, size); CFRelease (iconPath); } CFRelease (iconURL); } } } CFRelease (appBundle); } CFRelease (url); } CFRelease (pathCFString); } return hostIcon; } } // namespace juce