AfterEffect插件--常规功能开发--合成放大缩小--js脚本开发--AE插件

1.算法程序

  AfterEffect(AE)插件是Adobe公司开发的特效制作软件,稳定快速的功能和特效,在视频制作领域使用非常广泛,本文向大家介绍如何在项目里进行合成放大缩小功能。源代码如下所示:

{
    // Scale Composition.jsx
    // 
    // This script scales the active comp and all the layers within it.
    //
    // First, it prompts the user for a scale_factor, a new comp width, 
    // or a new comp height.
    // 
    // Next, it scales the comp and all the layers within it, including
    // cameras.
    
    function ScaleComposition(thisObj)
    {
        var scriptName = "Scale Composition";
        
        // This variable stores the scale_factor.
        var scale_factor = 1.0;
        var text_input = null;
        var scaleButton  = null;
        var widthButton  = null;
        var heightButton = null;
        
        
        function onScaleButtonClick()
        {
            this.parent.text_input.text = scale_factor;
        }
        
        
        function onWidthButtonClick()
        {
            var activeItem = app.project.activeItem;
            if ((activeItem == null) || !(activeItem instanceof CompItem)) {
                alert("Please select or open a composition first.", scriptName);
            } else {
                this.parent.text_input.text = Math.floor(activeItem.width * scale_factor);
            }
        }
        
        
        function onHeightButtonClick()
        {
            var activeItem = app.project.activeItem;
            if ((activeItem == null) || !(activeItem instanceof CompItem)) {
                alert("Please select or open a composition first.", scriptName);
            } else {
                this.parent.text_input.text = Math.floor(activeItem.height * scale_factor);
            }
        }
        
        
        function testNewScale(test_scale)
        {
            var is_ok = true;
            var activeItem = app.project.activeItem;
            if ((activeItem == null) || !(activeItem instanceof CompItem)) {
                alert("Please select or open a composition first.", scriptName);
            } else {
                if (test_scale * activeItem.width < 1 || test_scale * activeItem.width > 30000) {
                    is_ok = false;
                } else if (test_scale * activeItem.height < 1 || test_scale * activeItem.height > 30000) {
                    is_ok = false;
                }
            }
            
            return is_ok;
        }
        
        
        //
        // This function is called when the user enters text for the scale.
        //
        function on_textInput_changed()
        {
            var activeItem = app.project.activeItem;
            if ((activeItem == null) || !(activeItem instanceof CompItem)) {
                alert("Please select or open a composition first.", scriptName);
            } else {
                // Set the scale_factor based on the text.
                var value = this.text;
                if (isNaN(value)) {
                    alert(value + " is not a number. Please enter a number.", scriptName);
                } else {
                    var new_scale_factor;
                    if (this.parent.scaleButton.value == true) {
                        new_scale_factor = value;
                    } else if (this.parent.widthButton.value == true) {
                        new_scale_factor = value / activeItem.width;
                    } else {
                        new_scale_factor = value / activeItem.height;
                    }
                    if (testNewScale(new_scale_factor)) {
                        scale_factor = new_scale_factor;
                    } else {
                        alert("Value will make height or width out of range 1 to 30000. Reverting to previous value.", scriptName);
                        // Load text back in from current values.
                        if (scaleButton.value == true) {
                            onScaleButtonClick();
                        } else if (widthButton.value == true) {
                            onWidthButtonClick();
                        } else {
                            onHeightButtonClick();
                        }
                    }
                }
            }
        }
        
        
        function onScaleClick()
        {
            var activeItem = app.project.activeItem;
            if ((activeItem == null) || !(activeItem instanceof CompItem)) {
                alert("Please select or open a composition first.", scriptName);
            } else {
                // Validate the input field, in case the user didn't defocus it first (which often can be the case).
                this.parent.parent.optsRow.text_input.notify("onChange");
                
                var activeComp = activeItem;
                
                // By bracketing the operations with begin/end undo group, we can 
                // undo the whole script with one undo operation.
                app.beginUndoGroup(scriptName);
                
                // Create a null 3D layer.
                var null3DLayer = activeItem.layers.addNull();
                null3DLayer.threeDLayer = true;
                
                // Set its position to (0,0,0).
                null3DLayer.position.setValue([0,0,0]);
                
                // Set null3DLayer as parent of all layers that don't have parents.  
                makeParentLayerOfAllUnparented(activeComp, null3DLayer);
                
                // Set new comp width and height.
                activeComp.width  = Math.floor(activeComp.width * scale_factor);
                activeComp.height = Math.floor(activeComp.height * scale_factor);
                
                // Then for all cameras, scale the Zoom parameter proportionately.
                scaleAllCameraZooms(activeComp, scale_factor);
                
                // Set the scale of the super parent null3DLayer proportionately.
                var superParentScale = null3DLayer.scale.value;
                superParentScale[0] = superParentScale[0] * scale_factor;
                superParentScale[1] = superParentScale[1] * scale_factor;
                superParentScale[2] = superParentScale[2] * scale_factor;
                null3DLayer.scale.setValue(superParentScale);
                
                // Delete the super parent null3DLayer with dejumping enabled.
                null3DLayer.remove();
                
                app.endUndoGroup();
                
                // Reset scale_factor to 1.0 for next use.
                scale_factor = 1.0;
                if (this.parent.parent.optsRow.scaleButton.value) {
                    this.parent.parent.optsRow.text_input.text = "1.0";
                }
            }
        }
        
        
        // 
        // This function puts up a modal dialog asking for a scale_factor.
        // Once the user enters a value, the dialog closes, and the script scales the comp.
        // 
        function BuildAndShowUI(thisObj)
        {
            // Create and show a floating palette.
            var my_palette = (thisObj instanceof Panel) ? thisObj : new Window("palette", scriptName, undefined, {resizeable:true});
            if (my_palette != null)
            {
                var res = 
                    "group { \
                        orientation:'column', alignment:['fill','top'], alignChildren:['left','top'], spacing:5, margins:[0,0,0,0], \
                        introStr: StaticText { text:'Scale composition using:', alignment:['left','center'] }, \
                        optsRow: Group { \
                            orientation:'column', alignment:['fill','top'], \
                            scaleButton: RadioButton { text:'New Scale Factor', alignment:['fill','top'], value:'true' }, \
                            widthButton: RadioButton { text:'New Comp Width', alignment:['fill','top'] }, \
                            heightButton: RadioButton { text:'New Comp Height', alignment:['fill','top'] }, \
                            text_input: EditText { text:'1.0', alignment:['left','top'], preferredSize:[80,20] }, \
                        }, \
                        cmds: Group { \
                            alignment:['fill','top'], \
                            okButton: Button { text:'Scale', alignment:['fill','center'] }, \
                        }, \
                    }";
                
                my_palette.margins = [10,10,10,10];
                my_palette.grp = my_palette.add(res);
                
                // Workaround to ensure the edittext text color is black, even at darker UI brightness levels.
                var winGfx = my_palette.graphics;
                var darkColorBrush = winGfx.newPen(winGfx.BrushType.SOLID_COLOR, [0,0,0], 1);
                my_palette.grp.optsRow.text_input.graphics.foregroundColor = darkColorBrush;
                
                my_palette.grp.optsRow.scaleButton.onClick  = onScaleButtonClick;
                my_palette.grp.optsRow.widthButton.onClick  = onWidthButtonClick;
                my_palette.grp.optsRow.heightButton.onClick = onHeightButtonClick;
                
                // Set the callback. When the user enters text, this will be called.
                my_palette.grp.optsRow.text_input.onChange = on_textInput_changed;
                
                my_palette.grp.cmds.okButton.onClick = onScaleClick;
                
                my_palette.onResizing = my_palette.onResize = function () {this.layout.resize();}
            }
            
            return my_palette;
        }
        
        
        // 
        // Sets newParent as the parent of all layers in theComp that don't have parents.
        // This includes 2D/3D lights, camera, av, text, etc.
        //
        function makeParentLayerOfAllUnparented(theComp, newParent)
        {
            for (var i = 1; i <= theComp.numLayers; i++) {
                var curLayer = theComp.layer(i);
                var wasLocked = curLayer.locked;
                curLayer.locked = false;
                if (curLayer != newParent && curLayer.parent == null) {
                    curLayer.parent = newParent;
                }
                curLayer.locked = wasLocked
            }
        }
        
        
        //
        // Scales the zoom factor of every camera by the given scale_factor.
        // Handles both single values and multiple keyframe values.
        function scaleAllCameraZooms(theComp, scaleBy)
        {
            for (var i = 1; i <= theComp.numLayers; i++) {
                var curLayer = theComp.layer(i);
                if (curLayer.matchName == "ADBE Camera Layer") {
                    var curZoom = curLayer.zoom;
                    if (curZoom.numKeys == 0) {
                        curZoom.setValue(curZoom.value * scaleBy);
                    } else {
                        for (var j = 1; j <= curZoom.numKeys; j++) {
                            curZoom.setValueAtKey(j,curZoom.keyValue(j)*scaleBy);
                        }
                    }
                }
            }
        }
        
        
        // 
        // The main script.
        //
        if (parseFloat(app.version) < 8) {
            alert("This script requires After Effects CS3 or later.", scriptName);
            return;
        }
        
        var my_palette = BuildAndShowUI(thisObj);
        if (my_palette != null) {
            if (my_palette instanceof Window) {
                my_palette.center();
                my_palette.show();
            } else {
                my_palette.layout.layout(true);
                my_palette.layout.resize();
            }
        } else {
            alert("Could not open the user interface.", scriptName);
        }
    }
    
    
    ScaleComposition(this);
}

2.作者答疑


  如有疑问,请留言。

提示: 作者知了-联系方式1
提示: 作者知了-联系方式2

©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 219,928评论 6 509
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 93,748评论 3 396
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 166,282评论 0 357
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 59,065评论 1 295
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 68,101评论 6 395
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 51,855评论 1 308
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 40,521评论 3 420
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 39,414评论 0 276
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 45,931评论 1 319
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 38,053评论 3 340
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 40,191评论 1 352
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 35,873评论 5 347
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 41,529评论 3 331
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 32,074评论 0 23
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 33,188评论 1 272
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 48,491评论 3 375
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 45,173评论 2 357

推荐阅读更多精彩内容