diff --git a/.gitattributes b/.gitattributes new file mode 100644 index 0000000..dfe0770 --- /dev/null +++ b/.gitattributes @@ -0,0 +1,2 @@ +# Auto detect text files and perform LF normalization +* text=auto diff --git a/.github/workflows/build.yml b/.github/workflows/build.yml new file mode 100644 index 0000000..5764c8b --- /dev/null +++ b/.github/workflows/build.yml @@ -0,0 +1,19 @@ +on: push +name: Build my extension +jobs: + buildAIX: + name: Build AIX + runs-on: ubuntu-18.04 + steps: + - uses: actions/checkout@master + - name: Build AIX + id: build_aix + uses: ysfchn/appinventor-aix-action@master + with: + source: 'https://github.com/ysfchn/appinventor-sources.git' + - name: Upload Artifact + id: upload-artifact + uses: actions/upload-artifact@v1.0.0 + with: + name: ${{ steps.build_aix.outputs.file }} + path: appinventor-sources/appinventor/components/build/extensions/${{ steps.build_aix.outputs.file }} diff --git a/LICENSE b/LICENSE new file mode 100644 index 0000000..f96ce70 --- /dev/null +++ b/LICENSE @@ -0,0 +1,21 @@ +MIT License + +Copyright (c) 2020 Yusuf Cihan + +Permission is hereby granted, free of charge, to any person obtaining a copy +of this software and associated documentation files (the "Software"), to deal +in the Software without restriction, including without limitation the rights +to use, copy, modify, merge, publish, distribute, sublicense, and/or sell +copies of the Software, and to permit persons to whom the Software is +furnished to do so, subject to the following conditions: + +The above copyright notice and this permission notice shall be included in all +copies or substantial portions of the Software. + +THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR +IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, +FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE +AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER +LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, +OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE +SOFTWARE. diff --git a/README.md b/README.md new file mode 100644 index 0000000..8c63c37 --- /dev/null +++ b/README.md @@ -0,0 +1,8 @@ +# DynamicComponents-AI2 +Fully supported Dynamic Components extension for MIT App Inventor 2. It is based on Java's reflection feature, so it creates the components by searching for a class by just typing its name. So it doesn't have a limited support for specific components, because it supports every component which is ever added to App Inventor platform! + +## Blocks + +![](blockdescription.png) + +Source code is licensed under MIT license. diff --git a/blockdescription.png b/blockdescription.png new file mode 100644 index 0000000..998b6f8 Binary files /dev/null and b/blockdescription.png differ diff --git a/src/com/yusufcihan/DynamicComponents/DynamicComponents.java b/src/com/yusufcihan/DynamicComponents/DynamicComponents.java new file mode 100644 index 0000000..fafdb53 --- /dev/null +++ b/src/com/yusufcihan/DynamicComponents/DynamicComponents.java @@ -0,0 +1,232 @@ +package com.yusufcihan.DynamicComponents; + +import com.google.appinventor.components.annotations.*; +import com.google.appinventor.components.runtime.*; +import com.google.appinventor.components.common.*; +import com.google.appinventor.components.runtime.util.YailList; +import com.google.appinventor.components.runtime.errors.YailRuntimeError; + +import java.util.Hashtable; +import java.util.Iterator; +import java.util.ArrayList; +import java.util.List; +import java.lang.reflect.InvocationTargetException; +import java.lang.reflect.Constructor; +import java.lang.reflect.Method; +import java.lang.reflect.Parameter; +import java.lang.reflect.Modifier; +import java.util.Arrays; +import java.util.Set; +import java.lang.Boolean; + +import android.view.View; +import android.view.ViewGroup; + +@DesignerComponent(version = 3, + description = "Dynamic Components extension to create any type of dynamic component in any arrangement.

- by Yusuf Cihan", + category = ComponentCategory.EXTENSION, + nonVisible = true, + iconName = "https://yusufcihan.com/img/dynamiccomponents.png") +@SimpleObject(external = true) +public class DynamicComponents extends AndroidNonvisibleComponent implements Component { + + // Variables + private Hashtable COMPONENTS = new Hashtable(); + private List blacklist = Arrays.asList("CopyHeight", "CopyWidth", "wait", "onClick", "Column", "Row", "setLastWidth", "setLastHeight"); + private String BASE_PACKAGE = "com.google.appinventor.components.runtime"; + private String LAST_ID = ""; + + public DynamicComponents(ComponentContainer container) { + super(container.$form()); + } + + private String BasePackage() { + return BASE_PACKAGE; + } + + private void BasePackage(String packageName) { + BASE_PACKAGE = packageName; + } + + // ------------------------ + // MAIN METHODS + // ------------------------ + + @SimpleFunction(description = "Create a dynamic component that you want. It supports all components that added to App Inventor sources. Use 'in' parameter to specify the arrangement or canvas which new component will be placed in. Type the name of component in the 'componentName' section. (case sensitive) Use any component blocks to edit dynamic component's properties! ") + public void Create(AndroidViewComponent in, String componentName, String id) { + Object component = null; + LAST_ID = id; + // Check if id is used by another created dynamic component. + if (!COMPONENTS.containsKey(id)) + { + try + { + // Return the component class by looking the its name. + Class clasz = Class.forName(BASE_PACKAGE + "." + componentName); + // Create constructor object for creating a new instance. + Constructor constructor = clasz.getConstructor(new Class[] { ComponentContainer.class }); + // Create a new instance of specified component. + component = constructor.newInstance((ComponentContainer)in); + } + catch (Exception e) + { + // Throw a runtime error when something goes wrong. + throw new YailRuntimeError(e.getMessage(),"Error"); + } + COMPONENTS.put(id, component); + } + else + { + // Throw a runtime error when ID is already used for another component. + throw new YailRuntimeError("This ID is already used, please pick another.","Duplicate ID"); + } + + } + + @SimpleFunction(description = "Removes the component with specified ID from screen/layout and the component list. So you will able to use its ID again as it will be deleted.") + public void Remove(String id) { + Method m = null; + Object cmp = null; + // Don't do anything if id is not in the components list. + if (COMPONENTS.containsKey(id) == false) + return; + + try + { + // Get the component. + cmp = COMPONENTS.get(id); + // Remove its id from components list. + COMPONENTS.remove(id); + // Hide the component. + ((AndroidViewComponent)cmp).Visible(false); + } + catch (Exception eh) { } + } + + @SimpleFunction(description = "Returns last used ID.") + public String LastUsedID() { + return LAST_ID; + } + + @SimpleFunction(description = "Returns the component's itself for setting properties. Component needs to be created with Create block. Type an ID which you typed in Create block to return the component.") + public Object GetComponent(String id) { + return COMPONENTS.get(id); + } + + @SimpleFunction(description = "Returns the component type name.") + public String GetName(Object component) { + return component.getClass().getName(); + } + + @SimpleFunction(description = "Removes all created dynamic components. Same as Remove block, but for all created components.") + public void RemoveAll() { + Set keys = COMPONENTS.keySet(); + for(String key: keys){ + Remove(key); + } + } + + @SimpleFunction(description = "Get all available properties of a component which can be set from Designer as list along with types. Can be used to learn the properties of any component which is not static.") + public YailList GetDesignerProperties(Object component) { + // A list which includes designer properties. + ArrayList names = new ArrayList(); + // Get the component's class and return all methods from it. + Method[] methods = component.getClass().getMethods(); + for (Method mtd : methods) + { + // Read for @DesignerProperty annotations. + // So we can learn which method is used as property setter/getter. + if ((mtd.getDeclaredAnnotations().length == 2) && (mtd.isAnnotationPresent(DesignerProperty.class))) + { + // Get the DesignerProperty annotation. + DesignerProperty n = mtd.getAnnotation(DesignerProperty.class); + // Add editorType value and method name to the list. + names.add(YailList.makeList(new String[] { + mtd.getName(), + n.editorType() + })); + } + } + // Return the list. + return YailList.makeList(names); + } + + @SimpleFunction(description = "Set a property of a component by typing its name.") + public void SetProperty(Object component, String propertyName, Object propertyValue) { + // Read methods of the component. + Method[] methods = component.getClass().getMethods(); + // The method will be invoked. + Method method = null; + // Class for casting purpose. + Class caster = null; + try + { + for (Method mtd : methods) + { + // Check for one parametered (setter) method. + if((mtd.getName() == propertyName) && (mtd.getParameterCount() == 1)) + { + // Save it for later. + caster = mtd.getParameterTypes()[0]; + method = mtd; + break; + } + } + // Invoke the saved method. + method.invoke(component, propertyValue); + } + catch (Exception eh) + { + // Throw an error when something goes wrong. + throw new YailRuntimeError(eh.getMessage().toString(),"Error"); + } + } + + @SimpleFunction(description = "Get property value of a component.") + public Object GetProperty(Object component, String propertyName) { + // Read methods of the component. + Method[] methods = component.getClass().getMethods(); + // The method will be invoked. + Method method = null; + try + { + for (Method mtd : methods) + { + // Check for zero parametered (getter) method. + if((mtd.getName() == propertyName) && (mtd.getParameterCount() == 0)) + { + // Save it for later. + method = mtd; + break; + } + } + // Invoke the saved method and return its return value. + return method.invoke(component); + } + catch (Exception eh) + { + // Throw an error when something goes wrong. + throw new YailRuntimeError(eh.getMessage().toString(),"Error"); + } + } + + @SimpleFunction(description = "Returns the ID of component. Component needs to be created by Create block. Otherwise it will return -1.") + public String GetId(Object component) { + return getKeyFromValue(COMPONENTS, component); + } + + // ------------------------ + // PRIVATE METHODS + // ------------------------ + + // Getting key from value, found on: + // http://www.java2s.com/Code/Java/Collections-Data-Structure/GetakeyfromvaluewithanHashMap.htm + public String getKeyFromValue(Hashtable hm, Object value) { + for (Object o : hm.keySet()) { + if (hm.get(o).equals(value)) { + return (String)o; + } + } + return ""; + } +}